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内 容 提 要 


性 能 对 用 户 体验 有 着 至 关 重 要 的 影响 。 本 书 将 介绍 对 用 户 体验 产生 负面 影响 的 各 个 方面 ， 
并 概述 如 何 优化 iOS 应 用 的 性 能 。 全 书 共 5 个 部 分 ， 主 要 从 性 能 的 衡量 标准 、 对 应 用 至 关 重 要 
的 核心 优化 点 、iOS 应 用 开发 特有 的 性 能 优化 技术 以 及 性 能 的 非 代 码 方面 ， 讲 解 了 应 用 性 能 的 
优化 问题 。 本 书 的 主要 目的 是 展示 如 何 从 工程 学 的 角度 编写 最 优 代码 。 

本 书 适合 已 经 具有 Objective-C 和 iOS 实践 经 验 的 开发 人 员 阅 读 。 
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移动 互联 网 经 历 了 近 五 年 的 高 速 发 展 后 ， 增 长 速度 逐步 趋 于 平缓 。 疯 狂 过 后 ， 逐 渐 回 归 理 
性 。 尽 管 红利 不 再 ， 但 其 整体 规模 已 经 得 到 了 极 大 的 发 展 ， 手 机 和 App 已 经 成 为 了 人 们 日 
常生 活 中 很 重要 的 一 部 分 。 在 这 个 阶段 ， 移 动 互联 网 的 成 争 会 变 得 更 加 惨烈 。 冷 静 下 来 的 
人 们 不 禁 会 思考 一 个 问题 ， 如 何 让 我 们 的 App SUR 247)? 


排除 提供 服务 的 能 力 差异 (这 通常 与 App 本 身 无 关 )， 体 验 做 到 极致 才 更 有 机 会 捕获 用 户 
的 芳心 。 如 何 把 体验 做 得 更 好 呢 ? 一 方面 ,“ 颜 值 ”很 重要 ， 在 这 个 “看 脸 ” 的 年 代 ，App 
的 颜 值 太 差 一 定 会 被 用 户 嫌弃 。 另 一 方面 ， 性 能 更 重要 ， 更 高 的 性 能 意味 着 更 短 的 等 待 时 
间 、 更 平滑 流畅 的 体验 、 更 低 的 内 存 使 用 和 更 少 的 电量 消耗 。 对 于 每 个 程序 员 来 说 ,“ 我 
的 App 性 能 最 好 ”， 绝 对 是 一 件 值 得 炫 光 的 事情 。 但 要 想 真 正 做 到 这 一 点 ， 却 十 分 困难 。 


本 人 就 职 于 美 团 大 众 点 评 的 酒店 旅游 事业 群 ， 常 年 奋战 在 一 线 ， 专 注 于 iOS App 的 开发 和 
优化 ， 在 性 能 优化 方面 也 积累 了 大 量 的 实战 经 验 。 工 作 之 余 ， 一 直 有 这 样 一 种 想法 . 如 果 
有 一 本 书 能 够 系统 地 曾 述 iOS App 性 能 优化 的 方方面面 ， 一 定 会 对 我 和 我 们 的 团队 有 巨大 
的 帮助 ! 第 一 次 见 到 这 本 书 ， 我 就 立即 被 它 的 内 容 吸 引 了 。 书 中 涵盖 了 iOS App 性 能 优化 
的 方方面面 ， 既 有 广度 ， 又 有 深度 。 书 中 介绍 的 知识 点 ， 可 以 非常 容易 地 应 用 到 实际 的 项 
目 中 ; 很 多 的 技术 点 ， 和 我 们 之 前 所 做 的 优化 简直 是 不 谋 而 合 ， 大 有 相 见 恨 晚 之 感 。 本 书 
凝聚 了 作者 在 性 能 优化 方面 付出 的 大 量 心血 ， 值 得 每 一 位 期 望 进 阶 的 工程 师 深 入 地 阅读 和 
学 习 。 能 够 接手 本 书 的 翻译 任务 ， 对 我 来 说 既 兴 奋 又 充满 压力 。 尽 可 能 快 地 把 这 样 一 部 优 
秀 作 品 的 中 文 译 本 高 质量 地 交付 给 读者 ， 对 我 来 说 是 件 充满 激情 和 挑战 的 事情 。 

为 了 做 好 本 书 的 翻译 工作 ， 我 们 克服 了 许多 困难 。 首 先是 时 间 ， 我 们 需要 利用 业余 时 间 和 
尽 可 能 多 的 碎片 时 间 进 行 本 书 的 翻译 工作 ， 经 常 深夜 还 看 到 小 伙伴 们 仍然 在 奋 笔 疾 书 。 不 
仅 如 此 ， 我 还 有 些许 忧虑 ， 担 心 自己 把 握 不 好 原著 恰到好处 的 笔锋 ， 不 能 有 效 地 将 这 样 一 
部 优秀 的 作品 呈现 在 读者 面前 。 因 此 ， 我 们 对 这 次 翻译 格外 用 心 ， 与 几 位 合作 者 一 起 查阅 
了 大 量 相关 资料 ， 力 求 做 到 专业 词汇 准确 权威 ， 将 原 书 的 精华 呈现 给 每 一 位 读者 。 

现在 ， 我 怀 着 志 亚 的 心情 ， 将 此 译 闭 呈现 给 读者 ， 泡 望 得 到 读者 认可 ， 更 渴望 与 读者 成 为 
朋友 。 如 果 有 任何 问题 和 建议 ， 请 与 我 联系 (liangshixing@gmail.com)， 让 我 们 一 起 探讨 ， 
共同 进步 。 
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你 可 能 已 经 开发 过 一 个 令 人 赞叹 的 iOS 应 用 ， 或 者 正在 开发 。 如 有 果 应 用 整体 运行 良好 但 存 
在 一 些 缺 陷 ， 那 么 用 户 可 能 不 会 给 予 五 星 好 评 ， 这 些 缺 陷 甚 至 会 影响 发 布 。 

一 些 问 题 是 用 户 可 以 直接 发 现 的 ， 例 如 ， 用 户 点 击 表格 视图 的 某 一 项 时 出 现 抖动 现象 ， 应 
用 的 流量 消耗 过 多 或 耗 电量 巨大 。 但 是 这 些 问 题 可 能 发 生 在 更 深 的 层面 。 
优化 应 用 的 性 能 是 一 项 永 无 止境 的 工作 ， 尤 其 在 应 用 的 新 特性 、 操 作 系统 版 本 、 第 三 方 库 
和 设备 配置 层出不穷 的 情况 下 。 而 这 些 只 是 让 开发 者 关注 应 用 性 能 的 一 小 部 分 内 容 。 

一 项 研究 表明 ， 如 果 应 用 无 法 在 三 秒 内 加 载 启 动 ， 那 么 约 四 分 之 一 的 用 户 将 弃 用 此 应 用 ， 
约 三 分 之 一 的 用 户 会 将 这 段 令 人 不 快 的 经 历 转告 他 人 。 


用 户 希 望 应 用 运行 快速 、 响 应 迅速 且 不 占用 过 量 的 资源 。 本 书 将 介绍 对 用 户 体验 产生 负面 
影响 的 各 个 方面 ， 并 概述 如 何 优 化 应 用 的 性 能 。 


本 书 读者 


如 果 你 写 过 iOS 应 用 并 发 布 到 了 App Store， 那 么 你 的 隐 含 目标 是 让 应 用 更 好 、 更 快 、 更 流 
畅 ， 毫 无 疑问 ， 你 的 最 终 目标 是 让 应 用 为 用 户 所 喜爱 。 如 果 你 正在 寻找 实现 这 个 目标 的 方 
法 ， 那 么 本 书 正 是 为 你 而 准备 的 。 

你 应 该 已 经 具有 Objective-C 和 iOS 的 实践 经 验 。 虽 然 在 必要 时 为 内 容 完 整 起 见 会 介绍 一 
些 基 本 原理 ， 但 本 书 不 会 讨论 如 何 使 用 Objective-C 或 如 何 进行 iOS 的 入 门 开 发 。 
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55 — fX iOS 和 iPhone 1 F 2007 年 6 月 推出 。 在 早期 版 本 中 ， 开 发 人 员 忙 于 清理 代码 ， 为 
更 多 用 户 发 布 应 用 。 随 着 硬件 、 操 作 系 统 、 网 络 以 及 整体 生态 系统 的 不 断 提 升 ， 新 的 UI 
和 工程 设计 模式 不 断 涌现 ， 应 用 在 功能 、 稳 定性 和 性 能 方面 逐渐 成 熟 。 


通常 情况 下 ， 性 能 是 后 来 才 会 考虑 的 事情 。 从 某 种 程度 上 来 说 ,我 认同 这 种 观点 ， 毕 竞 最 












































































































































重要 的 是 先 完成 功能 ， 而 不 是 担心 性 能 。 在 软件 开发 周期 的 早期 考虑 性 能 通常 被 称 为 过 时 
优化 ， 但 是 ， 当 糟糕 的 性 能 表现 暴露 时 ， 问 题 就 大 严重 了 。 

本 书 的 主要 目的 是 向 读者 展示 如 何 从 工程 学 的 角度 编写 最 优 代 码 。 

本 书 并 非 通 过 计算 机 理论 科学 、 数 据 结构 和 算法 来 更 快 地 执行 程序 。 你 可 以 找到 很 多 关于 
这 些 主题 的 图 书 。 本 书 涵盖 了 实现 应 用 的 最 佳 实践 ， 即 使 在 非 理想 条 件 〈 低 存储 空间 、 不 
良 网 络 、 低 电量 等 ) 下 ， 让 用 户 仍然 可 以 有 效 地 使 用 应 用 ， 并 乐于 使 用 应 用 。 通 常 而 言 ， 
你 不 可 能 优化 所 有 的 参数 ， 但 考虑 有 效 因素 可 以 实现 最 佳 平衡 。 


书 预览 

本 书 预 览 

本 书 共 由 五 个 部 分 组 成 ， 每 一 个 部 分 由 一 章 或 多 章 根据 特定 的 主题 组 成 。 每 章 开 头 会 有 简 
短 的 摘要 说 明 。 

第 一 部 分 概述 如 何 衡 量 性 能 。 第 1 章 讨论 可 优化 的 方面 ， 并 概述 跟踪 应 用 性 能 时 需要 衡量 
的 参数 。 

第 二 部 分 回顾 对 应 用 至 关 重 要 的 核心 优化 点 。 第 2 章 讨论 内 存 管理 问题 ， 其 中 描述 了 内 存 
管理 模型 和 对 象 引 用 类 型 ， 还 讨论 了 影响 内 存 消耗 的 设计 模式 的 最 佳 实践 ， 即 单 例 和 依赖 
注入 。 第 3 章 讨论 电量 及 可 以 最 大 限度 减少 其 消耗 的 技术 。 第 4 章 为 并 发 编程 概述 ， 其 中 
描述 了 各 种 有 效 方法 ， 并 提供 了 对 比分 析 。 

第 三 部 分 涉及 iOS 应 用 开发 特有 的 性 能 优化 技术 。 第 5 章 深 入 探讨 应 用 的 生命 周期 ， 详 
细 介 绍 了 如 何 利用 生命 周期 事件 来 确保 资源 的 高 效 使 用 。 第 6 章 专 门 闸 述 针对 UL 的 优 
化 技术 。 第 7 章 和 第 8 章 分 别 讨论 网 络 和 数据 共享 。 第 9 章 深 入 探讨 了 应 用 的 安全 问题 ， 
了 解 增强 安全 性 会 对 应 用 的 运行 效率 产生 哪些 负面 影响 ， 以 及 如 何在 两 者 间 实 现 有 效 的 
平衡 。 
第 四 部 分 讨论 性 能 的 非 代码 方面 。 第 10 章 的 内 容 涉 及 测试 ， 特 别 是 性 能 测试 ， 此 外 还 讨 
论 了 持续 集成 和 测试 自动 化 。 第 11 章 概 述 开 发 过 程 中 用 于 衡量 性 能 的 工具 。 第 12 章 讨论 
埋 点 和 分 析 ， 以 及 如 何 从 生产 环境 的 应 用 中 收集 与 性 能 相关 的 数据 。 

第 五 部 分 重点 介绍 iOS 9 及 iOS 10。 第 13 章 概 述 iOS 9 的 变化 ， 并 从 性 能 角度 分 析 它 们 是 
如 何 影响 你 编写 的 代码 的 。 第 14 章 概述 了 iOS 10 的 变化 。 

本 书 提供 了 可 运行 的 代码 段 。 其 中 部 分 代码 段 可 以 原样 使 用 ， 或 只 需要 在 应 用 中 进行 小 修 
改 。 其 他 代码 段 可 能 需要 进一步 调整 ， 以 适应 你 自己 的 应 用 。 


每 章 还 提供 了 与 该 主题 相关 的 一 组 最 佳 实践 。 在 单一 应 用 中 可 能 无 法 始终 遵循 所 有 的 最 佳 
方案 ， 可 根据 应 用 的 具体 要 求 对 优化 点 进行 取舍 。 


在 线 资 源 
本 书 涉及 许多 在 线 博客 、 文 章 、 教 程 和 其 他 参考 资料 ， 并 在 合适 的 地 方 提 供 了 这 些 参考 文 
献 的 链接 。 如 果 你 发 现 有 遗漏 之 处 ， 请 随时 与 出 版 社 或 作者 联系 。 
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本 书 还 引用 了 几 个 应 用 的 截图 。 应 用 的 版 权 归 其 拥有 者 所 有 ， 本 书 使 用 屏幕 截图 仅 用 于 教 
育 和 说 明 。 


排版 约定 


本 书 使 用 了 下 列 排版 约定 。 


。 楷体 














表示 新 术语 或 重点 强调 内 容 。 


。 等 宽 字 体 (constant width) 





表示 程序 片段 ， 以 及 正文 中 出 现 的 变量 、 





和 关键 字 等 。 
。 加 粗 等 宽 字体 (constant width bold) 
表示 应 该 由 用 户 输入 的 命令 或 其 他 文本 。 
。 等 宽 斜 体 (Constant width italic) 
表示 应 该 由 用 户 输入 的 值 或 根据 上 下 文 确定 的 值 替 换 的 文本 。 

















该 图 标 表示 提示 或 建议 。 


该 图 标 表 示 一 般 注 记 。 


该 图 标 表示 警告 或 警示 。 


使 用 代码 示例 


补充 材料 (代码 示例 、 练 习 等 ) 可 以 从 https://github.com/gvaish/high-performance-ios-apps 


下 载 。 
本 书 是 要 








帮 你 完成 工作 的 。 一 般 来 说 ， 如 果 本 和 


函数 名 、 数 据 库 、 数 据 类 型 、 环 境 变量 、 语 名 








提供 了 示例 代码 ， 你 可 以 把 它 用 在 你 的 程 





序 或 文档 中 。 除 非 你 使 用 了 很 大 一 部 分 代码 ， 否 则 无 需 联 系 我 们 获得 许可 。 比 如 ， 用 本 书 
的 几 个 代码 片段 写 一 个 程序 就 无 需 获 得 许可 ， 销 售 或 分 发 OReilly 图书 的 示例 光盘 则 需要 
获得 许可 ， 引 用 本 书 中 的 示例 代码 回答 问题 无 需 获 得 许可 ， 将 书 中 大 量 的 代码 放 到 你 的 产 
品 文档 中 则 需要 获得 许可 。 








我 们 很 希望 但 并 不 强制 要 求 你 在 引用 本 书 内 容 时 加 上 引用 说 明 。 引 用 说 明 一 般 包 括 书 名 、 
作者 、 出 版 社 和 ISBN。 例 如 :“High Performance iOS Apps by Gaurav Vaish (O'Reilly). 
Copyright 2016 Gaurav Vaish, 978-1-491-91100-6.” 


如 果 你 觉得 自己 对 示例 代码 的 用 法 超出 了 上 述 许可 的 范围 ， 欢 迎 你 通过 permissions@ 
oreilly.com 与 我 们 联系 。 

















Safari? Books Online 





© Safari Books Online (http://www.safaribooksonline.com) 是 应 运 而 

< Safari 生 的 数字 图 书馆 。 它 同时 以 图 书 和 视频 的 形式 出 版 世界 顶级 技术 

和 商务 作家 的 专业 作品 。 技 术 专 家 、 软 件 开发 人 员 、Web 设计 师 、 

商务 人 士 和 创意 专家 等 ， 在 开展 调研 、 解 决 问 题 、 学 习 和 认证 培训 时 ， 都 将 Safari Books 
Online 视 作 获取 资料 的 首选 渠道 。 

对 于 组 织 团 体 、 政 府 机 构 和 个 人 ，Safari Books Online 提供 各 种 产品 组 合 和 灵活 的 定 

价 策略 。 用 户 可 通过 一 个 功能 完备 的 数据 库 检 索 系 统 访 问 O'Reilly Media, Prentice 

Hall Professional, Addison-Wesley Professional, Microsoft Press, Sams, Que, Peachpit 











Press, Focal Press, Cisco Press, John Wiley & Sons, Syngress, Morgan Kaufmann, IBM 
Redbooks, Packt, Adobe Press, FT Press, Apress, Manning. New Riders, McGraw-Hill, 
Jones & Bartlett, Course Technology 以 及 其 他 几 十 家 出 版 社 的 上 千 种 图 书 、 培 训 视 频 和 正 
式 出 版 之 前 的 书稿 。 要 了 解 Safari Books Online 的 更 多 信息 ， 我 们 网 上 见 。 


联系 我 们 
请 把 对 本 书 的 评价 和 问题 发 给 出 版 社 。 


美国 : 
O'Reilly Media, Inc. 
1005 Gravenstein Highway North 
Sebastopol, CA 95472 
中 国 : 
北京 市 西城 区 西直门 南大 街 2 号 成 铭 大 厦 C HE 807 (100035) 
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第 一 部 分 
开始 





我 们 将 在 第 一 部 分 定义 移动 应 用 的 性 能 ， 并 明确 性 能 指标 ， 即 那些 能 够 影响 整体 用 户 体验 
和 应 用 评分 的 因素 。 我 们 还 将 指出 一 些 可 测量 的 应 用 指标 ， 它 们 应 该 被 监测 并 随 着 时 间 持 
续 改 进 。 


























第 1 章 


移动 应 用 的 性 能 





本 书 假设 你 是 iOS 开发 人 员 ， 有 长 期 开发 原生 10S 应 用 的 经 验 ， 并 且 希 望 能够 从 众人 中 脱 
颖 而 出 ， 跻 身 于 顶尖 开发 人 员 之 列 。 

参考 以 下 统计 数据 。， 

。 应 用 首次 工作 出 错 以 后 ，79% 的 用 户 只 会 再 重 试 一 两 次 。 

。 当 应 用 载 入 时 间 超 过 3 秒 时 ，25% 的 用 户 会 放弃 使 用 该 应 用 。 

。 3196 的 用 户 会 将 糟糕 的 体验 转告 他 人 。 

这 些 数据 强调 了 性 能 对 应 用 的 重要 性 。 应 用 能 否 被 用 户 所 认可 不 仅仅 取决 于 其 功能 ， 还 取 
决 于 当 与 用 户 交 互 时 ， 应 用 能 耕 提 供 流 畅 的 体验 。 

几乎 完成 任意 特定 任务 的 应 用 都 能 在 App Store 中 找到 大 量 的 替代 品 。 但 用 户 只 会 坚持 使 
用 其 中 的 某 一 款 。 被 选中 的 这 一 款 要 么 无 可 取代 ， 要 么 极 少 出 现 故障 且 性 能 格外 出 众 。 

性 能 会 受 许多 重要 因素 所 影响 ， 这 些 因素 包括 内 存 消耗 、 网 络 带 宽 效 率 以 及 用 户 界面 的 响 
应 速度 。 我 们 先 概述 不 同类 型 的 性 能 特征 ， 然 后 再 对 它们 进行 测量 。 


mea Ab 
1.1 定义 性 能 
从 技术 视角 严格 来 说 ， 性 能 是 非常 模糊 的 术语 。 当 一 个 人 说 “这 是 个 高 性 能 的 应 用 ”时 ， 
其 实 我 们 无 从 判断 他 说 的 是 什么 。 他 是 说 应 用 消耗 的 内 存 少 ?应 用 节约 了 网 络 流量 ?还 是 
说 应 用 使 用 起 来 非常 流畅 ”总 而 言 之 ， 高 性 能 有 着 多 重 的 含义 和 丰富 的 解释 方式 。 
























































iE 1: Hewlett Packard Enterprise Software Solutions, “3 keys to a 5-star mobile experience” .(http://www. 





slideshare.net/HPESoftwareSolutions/3-keystoa5starmobileexperience) 





一 个 关注 点 ，( 实 际 上 收集 数据 的 ) 测量 是 另 一 个 关注 点 。 
我 们 将 在 第 11 章 深入 探索 测量 的 过 程 。 提 高 工程 参数 的 使 用 率 是 本 书 第 二 部 分 和 第 三 部 
分 的 重点 难题 。 


1.2 性 能 指标 
性 能 指标 是 面向 用 户 的 各 种 属性 。 每 个 属性 可 能 是 一 个 或 多 个 可 测量 工程 参数 的 一 个 要 素 。 


1.2.1. At 

内 存 涉及 运行 应 用 所 需 的 RAM 最 小 值 ， 以 及 应 用 消耗 的 内 存 平 均值 和 峰值 。 最 小 内 存 值 
会 严重 限制 硬件 ， 而 更 高 的 内 存 平 均值 和 峰值 意味 着 更 多 的 后 台 应 用 会 被 强制 关闭 。 

同时 还 要 确保 没有 泄漏 内 存 。 随 时 间 流 挝 而 持续 增长 的 内 存 消耗 意味 着 ， 应 用 很 可 能 会 因 
为 内 存 不 足 的 异常 而 月 涡 。 

我 们 会 在 第 2 章 中 对 内 存 进 行 深 入 讨论 。 


1.2.2 电量 消耗 

在 编写 高 性 能 代码 时 ， 电 量 消 耗 是 一 个 需要 重点 处 理 的 重要 因素 。 就 执行 时 间 和 CPU Tt 
源 的 利用 而 言 ， 我 们 不 仅 要 实现 高 效 的 数据 结构 和 算法 ， 还 需要 考虑 其 他 的 因素 。 如 果 某 
个 应 用 是 个 电 字 黑洞 ， 那 么 一 定 不 会 有 人 喜欢 它 。 

量 消耗 不 仅仅 与 计算 CPU 周期 有 关 ， 还 包括 高 效 地 使 用 硬件 。 除 了 要 实现 电量 消耗 最 
小 化 ， 还 要 确保 不 会 影响 用 户 体 验 。 


我 们 将 在 第 3 章 讨 论 这 个 问题 。 


1.2.8 初始 化 时 间 

应 用 在 启动 时 应 执行 刚好 够 用 的 任务 以 完成 初始 化 ， 从 而 满足 用 户 的 使 用 需求 。 执 行 这 些 
任务 消耗 的 时 间 就 是 应 用 的 初始 化 时 间 。 刚 好 够 用 是 一 个 开放 式 用 语 一 一 正确 的 平衡 点 取 
决 于 应 用 的 需要 。 

在 首次 使 用 应 用 时 创建 对 象 并 进行 初始 化 是 一 个 合理 的 选择 ， 例 如 ， 直 到 需要 使 用 对 象 时 
才 创 建 对 象 。 这 种 方式 被 称 为 惰性 初始 化 。 这 是 一 种 很 好 的 策略 ， 但 也 要 考虑 不 能 让 用 户 
总 是 在 执行 后 续 任 务 时 等 待 。 

下 面 列 举 了 你 可 能 想 在 应 用 初始 化 阶段 执行 的 一 些 动 作 ， 排 名 不 分 先后 。 

。 检查 应 用 是 否 为 首次 启动 。 

。 检查 用 户 是 否 已 经 登录 。 

。 如 果 用 户 已 经 登录 ， 尽 可 能 地 载 入 之 前 的 状态 。 

。 连接 服务 器 以 拉 取 最 新 的 变更 。 
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。 dH HAT UE EM, MRE, Yeu SEO UR I BEA UL 和 状态 。 
。 检查 是 否 存在 应 用 上 次 启动 时 挂 起 的 任务 ， 需 要 时 恢复 它们 。 

。 初始 化 后 续 需 要 使 用 的 对 象 和 线程 池 。 

。 初始 化 依赖 项 〈 如 对 象 关系 映射 、 崩 溃 报 告 系统 和 缓存 ) 。 


这 个 列表 可 能 会 迅速 变 长 ， 并 且 很 难 决定 哪些 条 目 一 定 要 在 启动 时 执行 ， 哪 些 可 以 延 后 几 
毫秒 再 执行 。 


我 们 将 在 第 5 章 探 讨 这 个 问题 。 


1.2.4 执行 速度 

一 旦 启动 应 用 ， 用 户 总 是 希望 它 可 以 尽 可 能 快 地 工作 。 一 切 必 要 的 处 理 都 应 该 在 尽 可 能 短 
的 时 间 内 完成 。 
例如 ， 在 照片 应 用 中 ， 用 户 通 常 希望 看 到 调整 亮度 或 对 比 度 等 简单 效果 的 实时 预览 效果 。 
因此 ， 相 应 的 处 理 需 要 在 儿 毫 秒 内 完成 。 


这 可 能 需要 本 地 计算 的 并 行 处 理 技术 或 能 够 将 复杂 任务 分 发 到 服务 器 。 我 们 将 在 第 4 章 和 
第 6 章 介绍 这 些 主题 ， 并 在 第 7 章 和 第 11 章 介绍 相关 工具 。 


1.2.5 ”响应 速度 


每 个 应 用 都 应 该 快速 地 响应 用 户 交互 。 在 应 用 中 所 做 的 一 切 优 化 和 权衡 最 终 都 应 该 体现 在 
响应 速度 上 。 


App Store 中 有 许多 应 用 可 以 完成 相似 或 相关 的 任务 。 这 为 用 户 提 供 了 很 大 的 选择 空间 ， 而 
用 户 基本 都 会 选择 响应 最 快 的 应 用 。 


第 4 章 介 绍 了 用 并 行 处 理 技术 优化 本 地 执行 。 第 5 章 和 第 6 章 介绍 了 实现 流畅 交互 的 最 佳 
实践 。 第 10 章 将 探索 如 何 对 应 用 进行 测试 。 


1.2.6 ”本 地 存储 

针对 任何 在 服务 器 上 存储 数据 或 通过 外 部 来 源 刷 新 数据 的 应 用 ， 开 发 人 员 应 该 对 本 地 存储 
的 使 用 有 所 规划 ， 以 便 应 用 有 具备 离线 浏 览 的 能 

例如 ， 用 户 都 希望 邮件 应 用 能 够 在 无 网 络 或 设备 离线 的 情况 下 浏览 历史 邮件 。 

同样 ， 新 闻 应 用 也 应 该 可 以 在 离线 模式 下 显示 最 近 更 新 的 新 闻 ， 并 标记 出 每 条 新 闻 是 否 已 读 。 
然而 ， 从 本 地 存储 中 载 入 和 同步 数据 应 该 迅速 、 便 捷 。 这 不 仅 需 要 选择 要 在 本 地 缓存 的 数 
据 和 要 优化 的 数据 结构 ， 还 需要 提供 一 系列 的 配置 选项 并 确定 数据 同步 的 频率 。 

如 果 你 的 应 用 使 用 了 本 地 存储 ， 那 么 请 提供 一 个 清除 数据 的 选项 。 遗 憾 的 是 ， 市 场 上 的 大 
部 分 应 用 都 没有 提供 此 选项 。 更 让 人 烦恼 的 是 ， 一 些 应 用 竟然 会 消 耗 数 百 兆 的 存储 空间 。 
用 户 会 频繁 地 和 伸 载 这 些 应 用 来 回收 本 地 存储 。 这 会 导致 糟糕 的 用 户 体验 ， 从 而 威胁 应 用 的 
成 功 。 
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如 图 1-1 所 示 ， 超 过 12GB 的 空间 已 经 被 使 用 了 ， 留 给 用 户 的 内 存 还 有 950MB. 
部 分 的 数据 可 以 安全 地 从 本 地 删除 。 这 些 应 用 应 该 提供 请 至 








其 实 ， 


缓存 的 选项 。 





《 General Usage 
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图 1-1: 磁盘 使 用 状况 





一 定 要 向 终端 用 户 提供 清空 本 地 缓存 的 选项 。 
如 果 用 户 开启 了 iCloud 的 备份 功能 ， 那 么 应 用 的 数据 将 会 消耗 用 户 的 存储 限 
额 ， 请 谨慎 使 用 。 





第 7 章 、 第 8 章 和 第 9 章 会 介绍 本 地 存储 相关 的 话题 。 


1.2.7 互 操作 性 


用 户 可 能 会 使 用 多 个 应 用 来 完成 某 个 任务 ， 这 就 需要 这 些 应 用 直接 提供 互 操作 的 能 力 。 例 
如 ， 一 个 相册 可 能 需要 一 个 幻灯 片 应 用 来 实现 最 佳 的 浏览 体验 ， 但 需要 另 一 个 应 用 来 编辑 





照片 。 其 中 浏览 照片 的 应 用 要 能 够 将 照片 发 送 到 编辑 器 ， 并 接收 编辑 后 的 图 片 。 
iOS 为 实现 应 用 间 的 互 操作 和 数据 共享 提供 了 多 种 机 制 ， 其 中 包括 UIActivityViewController, 


深层 链接 、MultipeerConnectivity 框架 ， 





[riores 





入 的 数据 时 还 要 注意 安全 隐患 。 
如 果 某 个 应 用 向 附近 设备 共享 数据 时 需要 花费 很 长 时 间 准 备 数据 ， 那 么 用 户 体验 就 会 非常 


糟糕 。 


我 们 会 在 第 8 章 中 讨论 这 些 内 容 。 


可 


为 深层 链接 定义 良好 的 URL 结构 与 编写 优异 的 代码 来 解析 URL 同样 重要 。 类 似 地 ， 使 用 
共享 对 话 框 共享 数据 时 ， 精 确 识别 用 于 分 享 的 数据 非常 重要 ， 同 时 ， 在 处 理 不 同 数据 源 传 




















1.2.8 网 络 环境 


移动 设备 会 在 不 同 网 络 环境 下 使 用 。 为 了 确保 能 够 提供 最 好 的 用 户 体验 ， 你 的 应 用 应 当 适 





应 各 种 网 络 条 件 : 


。 高 带宽 稳定 网 络 
。 低 带 宽 稳 定 网 络 
。 高 带宽 不 稳定 网 络 
。 低 带 宽 不 稳定 网 络 
。 无 网 络 




















为 用 户 提 供 进 度 指 示 或 错误 信息 是 相对 合理 的 方式 ， 无 尽 的 等 待 或 月 溃 则 让 人 无 法 接受 。 

图 1-2 的 屏幕 截图 展示 了 向 终端 用 户 传递 信息 的 不 同方 式 。TuneIn 应 用 显示 了 已 经 缓冲 的 
言 息 流 大 小 ， 以 此 告诉 终端 用 户 还 需要 等 待 多 和 久 才 可 以 播放 音乐 。MoneyControl 和 Bank 
of America 等 其 他 应 用 仅 提供 了 不 明确 的 进度 条 ， 这 在 非 流 式 应 用 中 是 更 为 常见 






































的 样式 。 
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Mobile Banking 


Bankof America > 











图 1-2: 因为 网 络 环境 差 或 数据 量 大 而 显示 的 不 同 提示 信息 
我 们 将 在 第 7 章 中 深入 探讨 此 话题 。 


1.2.9 ”带宽 


人 们 会 在 不 同 的 网 络 条 件 下 使 用 自己 的 移动 设备 ， 网 速 从 每 秒 数 千 字 市 到 每 秒 数 十 兆 字 
因此 ， 带 宽 的 优化 使 用 是 定义 应 用 质量 的 另 一 个 关键 参数 。 此 外 ， 在 高 





个 基于 低 带宽 网 络 开发 的 应 用 可 能 会 产生 完全 不 同 的 结果 。 








RES 


了 8 宽 网 络 下 运行 
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2010 年 左右 ， 我 和 我 的 团队 正在 印度 开发 一 款 应 用 。 由 于 处 于 低 带 宽 网 络 ， 应 用 的 本 地 初 
始 化 速度 要 比 从 服务 器 端 载 入 资源 快 得 多 ， 于 是 我 们 针对 这 种 情况 进行 了 优化 。 

然而 ， 当 这 款 应 用 投入 韩国 市 场 时 ， 我 们 对 它 进 行 了 测试 ， 结 果 却 让 人 大 跌眼镜 。 之 前 所 
进行 的 所 有 优化 几乎 毫 无 意义 ， 我 们 不 得 不 重 写 了 大 部 分 可 能 导致 资源 和 数据 冲突 的 相关 
代码 。 

为 提高 性 能 所 做 的 设计 并 非 每 次 都 能 如 愿 ， 也 可 能 会 导致 相反 的 效果 。 

第 7 章 包含 了 优化 使 用 带宽 的 最 佳 实践 。 


1.2.10 ”数据 刷新 


即使 没有 提供 离线 浏览 能 力 ， 你 仍然 可 以 从 服务 器 端 周期 性 地 刷新 数据 。 刷 新 的 频率 和 每 
次 传输 的 数据 量 将 决定 数据 传输 的 总 量 。 如 果 传 输 的 字 布 数 过 大 ， 那 用 户 必然 会 快速 耗 尽 
自己 的 流量 计划 。 当 流量 消耗 大 到 一 定 程度 时 ， 你 的 应 用 很 可 能 会 流失 用 户 。 

在 iOS 6.x 或 更 低 版 本 中 ， 在 后 台 运 行 的 应 用 不 能 刷新 数据 。 从 iOS 7 开始 ， 应 用 可 以 在 后 
台 周 期 性 地 刷新 数据 。 对 于 在 线 聊天 类 应 用 ， 持 和 久 的 HTTP 连接 或 原生 TCP 连接 可 能 会 非 
常 有 用 。 


第 5 章 和 第 7 章 会 介绍 这 部 分 内 容 。 


1.2.11 多 用 户 支 持 

家 庭 成 员 间 可 能 会 共享 移动 设备 ， 或 者 一 个 用 户 可 能 会 拥有 同一 应 用 的 多 个 账号 。 例 如， 
兄弟 姐妹 间 可 能 会 共享 一 个 iPad 来 玩 游戏 。 再 比如 ， 家 庭 成 员 可 能 会 在 旅游 时 配置 一 个 设 
备 来 查收 全 家 人 的 电子 邮件 ， 以 减少 漫游 费用 ， 尤 其 是 在 境外 旅游 时 。 类 似 地 ， 一 个 人 也 
可 能 会 配置 多 个 电子 邮件 账号 。 

是 否 支 持 多 个 并 发 用 户 取 决 于 产品 的 需要 。 一 旦 决定 提供 此 类 功能 ， 请 参考 以 下 准则 。 

。 添加 新 用 户 应 尽 可 能 高 效 。 

。 在 不 同 用 户 之 间 更 新 应 尽 可 能 高 效 。 

。 在 不 同 用 户 之 间 切 换 应 尽 可 能 高 效 。 

。 用 户 数据 的 界限 应 该 简洁 且 没 有 bug. 

图 1-3 展示 了 两 个 提供 了 多 用 户 支 持 的 应 用 。 左 边 和 右边 分 别 展示 了 Google 和 Yahoo 应 
用 的 账号 选择 功能 。 
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9 
gh 
& 1-3: Google 和 Yahoo 应 用 都 提供 了 多 用 户 支持 


























第 9 章 将 介绍 如 何在 应 用 支持 多 用 户 的 同时 保障 安全 以 及 其 他 内 容 。 


1.212 Hub xx 


如 果 你 已 经 创建 了 多 个 允许 或 需要 登录 的 应 用 ， 那 么 支持 单 点 登录 (single sign-on, SSO) 
是 非常 棒 的 选择 。 如 果 用 户 登录 了 一 个 应 用 ， 只 需要 点 击 一 次 ， 就 可 以 登录 到 其 他 的 应 
用 中 。 


这 个 过 程 不 仅 需要 支持 跨 应 用 的 数据 共享 ， 还 需要 分 享 状 态 、 跨 应 用 同步 等 。 例 如 ， 如 果 
用 户 注销 了 其 中 某 个 应 用 ， 则 通过 SSO 登录 的 所 有 其 他 应 用 也 应 能 注销 掉 。 


此 外 ， 应 用 之 间 的 同步 应 该 是 安全 的 。 


第 9 章 将 会 介绍 这 部 分 内 容 。 


1.243 ”安全 

安全 对 移动 应 用 来 说 是 最 重要 的 ， 因 为 敏感 信息 可 能 会 在 应 用 间 共 享 。 因此， 对 所 有 通信 
以 及 本 地 数据 和 共享 数据 进行 加 密 就 显得 尤为 重要 了 。 

实现 安全 需要 更 多 的 计算 、 内 存 和 存储 ， 但 这 与 最 大 化 运行 速度 、 最 小 化 内 存 和 存储 使 用 
的 目标 相 冲 突 。 
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因此 ， 你 需要 在 安全 和 其 他 因素 之 间 进 行 权衡 。 


引入 多 个 安全 层 会 影响 性 能 ， 并 对 用 户 体验 造成 可 感知 的 负面 影响 。 如 何 设 定安 全 的 基线 
需要 参考 对 用 户 群 体 的 统计 分 析 。 此 外 ， 硬 件 在 其 中 扮演 了 重要 的 角色 : 选择 会 因为 不 同 
设备 的 计算 能 力 而 有 所 不 同 。 


第 9 章 将 会 深入 介绍 安全 。 








1.2.44 Hae 

应 用 可 能 会 而 且 确 实 会 月 沉 。 过 度 优 化 会 导致 月 沉 。 同 样 ， 使 用 原始 C 代码 也 可 能 会 导致 
HAL 

高 性 能 的 应 用 不 仅 应 尽 可 能 地 人 避免 月 涡 ， 还 应 该 在 月 涡 发 生 时 优雅 地 恢复 ， 尤 其 是 在 进行 
某 个 操作 的 过 程 中 发 生 朋 涡 时 。 

第 12 章 会 深入 讨论 月 涡 报 告 、 检 测 和 分 析 。 


1.3 应 用 性 能 分 析 


我 们 在 前 面 讨 论 过 一 些 参数 ， 通 过 测量 它们 来 分 析 应 用 的 方式 有 两 种 : 采样 和 埋 点 。 接 下 
来 我 们 将 逐一 介绍 。 


1.3.1 采样 


顾名思义 ,采样 (或 基于 探测 点 的 性 能 分 析 ) 是 指 以 一 定 的 周期 间隔 采集 状态 ， 这 通常 需 
要 借助 工具 。 我 们 将 在 11.2 节 中 介绍 这 些 工 具 。 由 于 不 会 干扰 应 用 的 执行 ， 因 此 采样 可 以 
很 好 地 提供 应 用 的 全 景 图 。 采 样 的 不 足 之 处 在 于 它 不 能 返回 100% 精确 的 细节 。 如 有 果 采 样 
的 频率 是 10 毫秒 ， 那 么 你 就 无 法 得 知 在 探测 点 之 间 的 9.999 毫秒 内 发 生 了 什么 。 






































采样 可 以 作为 初始 的 性 能 调研 手段 ， 并 可 用 于 跟踪 CPU 和 内 存 的 使 用 情况 。 


1.3.2 HA 

通过 修改 代码 ， 记 录 细 市 信息 的 埋 点 能 够 提供 比 采 样 更 加 精确 的 结果 。 你 既 可 以 在 关键 部 
分 主动 埋 点 ， 也 可 以 在 性 能 分 析 或 处 理 用 户 反 馈 时 有 针对 性 地 埋 点 ， 以 便 解 决 问题 。1.4.3 
市 将 深入 讨论 这 一 过 程 。 











因为 埋 点 需要 注入 额外 代码 ， 所 以 它 一 定 会 影响 应 用 的 性 能 ， 对 内 存 或 速度 
(或 同时 对 二 者 ) 造成 损害 。 

















1.4 测量 
现在 ， 我 们 已 经 确定 了 需要 测量 的 参数 ， 并 且 研 究 了 测量 所 需要 的 不 同类 型 的 分 析 。 我 们 
先 简单 了 解 一 下 如 何 实现 测量 。 
通过 测量 性 能 并 找 出 真正 存在 问题 的 地 方 ， 你 可 以 避免 掉 入 过 早 优化 的 陷阱 。 高 德 纳 曾经 
这 样 描述 过 早 优化 : 

真正 的 问题 在 于 ,程序 开发 人 员 为 提升 程序 效率 在 错误 的 方向 和 时 间 点 浪费 了 大 

多 时 间 ; 过 早 优化 是 编程 领域 的 万 恶 (至 少 是 绝 大 多 数 的 恶 ) 之 源 。? 


1.4.1 设置 工程 与 代码 
接 下 来 ， 我 们 将 建立 一 个 工程 ， 以 便 在 开发 和 生产 阶段 测量 已 经 定义 好 的 参数 。 针 对 工程 
配置 、 安 装 和 代码 实现 共有 三 类 任务 。 
。 构建 与 发 布 
人 确保 能 够 轻松 地 构建 和 发 布 应 用 。 
。 可 测试 性 
确保 你 的 代码 能 够 同时 在 模拟 数据 和 真实 数据 之 上 工作 ， 其 中 包括 能 够 模拟 真实 场景 的 
隔离 环境 。 
。 可 跟踪 性 
确保 你 能 够 通过 明确 问题 发 生 的 位 置 和 代码 行为 来 处 理 错误 。 
接 下 来 将 逐一 讨论 这 些 设 置 项 。 
1. 构建 与 发 布 
构建 和 发 布 是 直到 最 近 才 出 现 的 话题 。 好 在 由 于 对 灵活 和 敏捷 的 强烈 需求 ， 系 统 和 工具 得 
到 了 改进 。 改 进 后 的 系统 和 工具 现在 可 以 加 速 拉 取 依赖 信息 ， 加 速 构建 和 发 布 用 于 测试 或 
企业 分 发 的 产品 ， 也 可 以 为 公众 发 布 而 提高 提交 文件 到 iTunes Connect 的 速度 。 
2000 年 ，Joel Spolsky 在 其 发 表 的 一 篇 博文 (http://www.joelonsoftware.com/articles/fog0000000043. 
html) 中 提出 了 一 个 回 题 :“ 你 能 〈 从 源码 ) 一 键 构建 自己 的 应 用 吗 ? ”这 个 问题 现在 依 
然 成 立 ， 且 问题 的 答案 很 可 能 会 决定 你 在 发 现 缺 陷 或 瓶颈 后 改进 质量 和 解决 性 能 问题 的 
速度 。 
基于 Ruby 语言 实现 的 CocoaPods (https://cocoapods.org) 实际 上 是 Objective-C 和 Swift T. 
程 的 依赖 管理 器 。 “CocoaPods 与 Xcode 命令 行 工具 相 集 成 ， 可 用 于 构建 与 发 布 。 
2. 可 测试 性 
每 个 应 用 都 包含 多 个 协同 工作 的 组 件 。 一 个 设计 良好 的 系统 应 该 遵循 低 契 合 和 高 内 采 ， 并 


























































































































注 2: Donald Knuth, “Computer Programming as an Art" (https://en.wikiquote.org/wiki/Donald_Knuth#Computer_ 
Programming as an Art .281974.29) 

iE 3: 在 撰写 本 书 时 ， 以 CocoaPods 发 布 的 绝 大 多 数 对 象 都 是 用 Objective-C 编写 的 ， 毕 竟 Swift LE Objective-C 
要 新 得 多 。 
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fo VER TEE E ABBA PER CHR 

可 以 通过 模拟 依赖 项 目 对 每 个 组 件 进行 隔离 测试 。 一 般 来 说 ， 测 试 有 两 种 类 型 。 

。 单元 测试 
验证 每 个 代码 单元 在 隔离 环境 下 的 操作 。 常 见 的 做 法 是 ， 在 特定 的 环境 中 用 不 同 的 输入 
数据 反复 地 调用 一 些 方法 ， 以 评估 代码 的 表现 。 

。 功能 测试 
验证 组 件 在 最 终 集成 的 安装 包 中 的 操作 。 可 以 在 软件 的 最 终 发 布 版 本 中 验证 ， 也 可 以 在 
某 个 为 测试 而 构建 的 参考 应 用 中 验证 。 

我 们 将 在 第 10 章 中 深入 讨论 测试 。 

3. 可 跟踪 性 

在 开发 阶段 ， 埋 点 可 以 帮助 我 们 确定 性 能 优化 的 优先 级 、 提 高 对 问题 现场 的 还 原 能 力 ， 并 

提供 更 多 的 调试 信息 。 崩 溃 报 告 专注 于 从 软件 的 产品 版 本 中 收集 调试 信息 。 


1.4.2 ”设置 崩溃 报告 

崩 涡 报告 系统 收集 用 于 分 析 应 用 的 调试 日 志 。 市 面 上 有 数 十 种 崩 涡 报告 系统 。 本 书 选 用 了 
Flurry (http://www.furry.com)， 但 这 并 不 代表 我 对 其 他 系统 有 任何 偏见 。 选 用 Flurry 的 主要 
原因 是 ， 只 用 一 个 SDK 就 可 以 同时 实现 崩溃 报告 和 埋 点 。 我 们 将 在 第 12 章 深 入 介绍 埋 点 。 
要 想 使 用 Flurry， 需 要 在 www.furry.com 中 建立 一 个 账户 ， 得 到 一 个 API 密 钥 ， 然 后 下 载 
并 设置 Flurry SDK。 例 1-1 展示 了 初始 化 Flurry 的 代码 。 


例 1-1 在 应 用 委托 中 配置 崩溃 报告 


#import "Flurry.h" 






































- (BOOL)application: (UIApplication *)application 
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { 


[Flurry setCrashReportingEnabled:YES]; 


[Flurry startSession:Q"API KEY"]; @ 
} 


@ 用 账户 关联 的 API 密 钥 (可 以 在 Flurry 的 Dashboard 中 找到 ) FH API KEY, 


崩溃 报告 系统 会 用 NSSetUncaughtExceptionHandler 方法 设置 全 局 的 异常 处 
理 器 。 使 用 自 定义 的 处 理 器 则 会 失效 。 

如 果 和 希望 继续 使 用 自己 的 处 理 器 ， 那 么 你 需要 在 崩溃 报告 系统 初始 化 之 后 再 
进行 设置 。 此 外 ， 还 可 以 通过 NSGetUncaughtExceptionHandler 方法 得 到 月 
涡 报 告 系统 所 设置 的 处 理 器 。 




















1.4.8 ”对 应 用 埋 点 


对 应 用 进行 埋 点 是 了 解 用 户 行为 的 一 个 重要 步骤 ， 但 更 重要 的 目的 是 识别 应 用 的 关键 路 
径 。 注 入 特定 的 代码 以 记录 关键 指标 是 提升 应 用 性 能 的 重要 步骤 。 
对 依赖 进行 抽象 化 和 封装 是 个 好 主意 。 这 样 就 可 以 在 最 后 再 进行 切换 ， 甚 至 
可 以 在 作出 最 终 决 定之 前 同时 使 用 多 个 系统 。 这 在 项 目 处 于 评估 阶段 且 存 在 
多 个 备 选 方案 时 尤为 有 用 。 





















































如 例 1-2 所 示 ， 我 们 将 增加 一 个 名 为 HPInstrumentation 的 类 来 封装 埋 点 。 现 在 ， 我 们 用 
NSLog 登录 控制 台 ， 并 向 服务 嚣 发送 细 市 。 
例 1-2 HPInstrumentation 类 封装 了 埋 点 SDK 


//HPINstrumentation.h 
@interface HPInstrumentation : NSObject 





*(void)logEvent:(NSString *)name; 
*(void)logEvent:(NSString *)name withParameters:(NSDictionary *)parameters; 


Qend 


/ | HPInstrumentation.m 
(implementation HPInstrumentation 


*(void)logEvent:(NSString *)name 


NSLog(@"%@", name); 
[Flurry logEvent:name]; 


*(void)logEvent:(NSString *)name withParameters:(NSDictionary *)parameters 


NSLog(@"%@ -» %@", name, params); 
[Flurry logEvent:name withParameters:parameters]; 


} 
@end 
先 在 应 用 生命 周期 的 三 个 关键 阶段 进行 埋 点 〈 见 例 1-3) : 
。 每 当 应 用 进入 前 人 台 ，applicationDidBecomeActive: 方法 会 被 调用 
。 每 当 应 用 进入 后 人 台 ，applicationDidEnterBackground: 方法 会 被 调用 
。 如 果 应 用 收 到 低 内 存 警 告 ，applicationDidReceiveMemoryWarning: 方法 会 被 调用 
出 于 娱乐 的 目的 ， 我 们 在 HPFirstViewController 中 添加 一 个 按钮 ， 点 击 该 按钮 将 导致 应 用 崩溃 。 
例 1-3 在 应 用 委托 中 的 基础 埋 点 


- (void)applicationDidBecomeActive:(UIApplication *)application 
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[HPInstrumentation logEvent:Q"App Activated"]; 


- (void)applicationDidEnterBackground:(UIApplication *)application 
{ 


} 


[HPInstrumentation logEvent:@"App Backgrounded"]; 


- (void)applicationDidReceiveMemoryWarning:(UlApplication *)application 


[HPInstrumentation logEvent:Q"App Memory Warning"]; 
} 
我 们 将 这 些 事件 分 别 命 名 为 App Activated, App Backgrounded 和 App Memory Warning, {R 
可 以 选择 自己 喜欢 的 任意 名 称 ， 也 可 以 使 用 数字 。 


埋 点 不 应 该 取代 日 志 。 日 志 可 以 非常 详细 。 但 因为 向 服务 器 报告 时 会 消耗 网 
络 资源 ， 所 以 你 应 该 尽 可 能 少 地 埋 点 。 

因此 ， 只 对 你 和 其 他 工程 或 产品 团队 的 成 员 感 兴趣 的 事件 进行 埋 点 是 非常 重 
要 的 。( 这 些 事 件 要 包含 足够 多 的 数据 以 满足 重要 报告 的 需要 。) 

埋 点 和 过 度 埋 点 之 间 并 没有 清晰 的 分 界线 。 一 开始 应 仅 对 少量 报告 进行 地 
点 ， 然 后 随 着 时 间 的 推进 逐步 增加 埋 点 的 覆盖 率 。 




















L 
HH 








接 下 来 添加 一 个 UI 控件， 让 它 能 够 触发 一 个 月 涡 ， 这 样 我 们 就 可 以 看 到 月 涡 报告 
图 14 展示 了 崩溃 按钮 的 UI。 例 1-4 展示 了 连接 Touch Up Inside 事件 和 crashButtonWasClicked: 
方法 的 代码 。 











Referencing Outlets 
CerashButton )- (X First View Cont... 
New Referencing Outlet 








Referencing Outlet Collections 
New Referencing Outlet Collection. 


i | <> | BB vCircle) @v) B v & v Bl r) OF) [| View) |. | Button - Generate Crash | 4 À >| BBsew?s oO 
> E Tab Bar Controller Scene (=e | | Triggered Segues 
action Oo 
Y [S] First View Controller - Feed Scene © Outlet Collections 
v ( First View Controller - Feed gestureRecognizers. o 
(5) Top Layout Guide Sent Events 
H) Bottom Layout Guide D End OR ERE o 
Editing Changed fo) 
View à " Editing Did Begin o 
|... Label - First View F t V Editing Did End [o] 
HH Button - Generate Crash IrSt View ea 9 
> 图 Constraints Touch Down (9) 
X Tab Bar Item - Feed Touch Down Repeat o 
@ First Responder Touch Drag Enter Oo 
ToC METRO Touch Drag Exit [6] 
Exit o o o Touch Drag Inside o 
Generate Crasho Touch Drag Outside. o 
> E Second View Controller - Photos... a H Tash Ug iii aarti 
crashButtonWa... 
> Touch Up Outside o 
Value Changed O 
OJ 
O 
o 











e n uu em 
一 View Controller - A controller that 
" (Hli As Imi iri m)(a ls 19] supports the fundamental view- 
© ) management model in iPhone OS. 














m m Hg c Ł £ |NoSelection 











1-4; 在 故事 板 中 添加 Generate Crash 按钮 





例 1-4 抛 出 异常 使 得 应 用 崩溃 


- (IBAction)crashButtonWasClicked: (id)sender 


[NSException raise:@"Crash Button Was Clicked" format:Q""]; 


} 
让 我 们 与 应 用 进行 交互 ， 从 而 产生 一 些 事件 。 
(1) 安装 并 运行 应 用 。 
(2) 将 应 用 切换 到 后 台 。 
(3) 将 应 用 切换 到 前 台 。 
(4) 多 次 重复 第 2 步 和 第 3 步 。 
(5) 点 击 Generate Crash 按钮 ， 导 致 应 用 崩溃 。 
(6) 再 次 运行 应 用 。 直 到 此 时 崩溃 报 告 才 会 实际 发 送 到 服务 器 。 
第 一 批 埋 点 事件 和 崩溃 报告 会 在 稍 后 上 报到 服务 器 并 得 到 处 理 。 报 告 可 能 在 一 段 时 间 后 才 
能 在 Flurry 的 Dashboard 中 出 现 。 然 后 你 可 以 进入 Dashboard AA ix e SF (FRI RA 1 Tf dr e 
你 应 该 会 看 到 与 下 面 截 图 类 似 的 内 容 ， 以 下 截图 是 为 我 的 应 用 从 Dashboard 中 截取 的 。 
图 1-5 展示 了 用 户 会 话 ， 用 户 会 话 指 的 是 有 多 少 用 户 在 一 天 中 至 少 启动 了 应 用 一 次 。 多 次 
运行 可 能 被 认定 是 同一 个 会 话 ， 也 可 能 不 会 被 认定 是 同一 个 会 话 ， 这 主要 取决 于 多 次 运行 
之 间 的 时 间 间 隔 。 



























































Sessions v Expan(@ Download cev I 


Zoom: days | weeks | months 














Number of Sessions. 














1-5: 用 户 会 话 报告 








图 1-6 展示 了 每 个 埋 点 事件 的 详细 崩溃 情况 。 这 份 报告 更 加 有 用 ， 因 为 它 可 以 让 我 们 深 入 
了 解 应 用 的 使 用 率 。 比 如 ， 它 精确 地 指出 了 应 用 中 的 哪些 部 分 比 其 他 部 分 使 用 频 度 更 高 。 
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Event Summary Statistics Explain(2) Download CSV } 
Event Events per 
Total Event Occurrences ~ Unique Event Session (Daily 

Event Name Occurrences (Daily Avg) Users (Daily Avg) Avg) Analyses 

App_Activate 240 8.28 0.62 0.95 d odi filter 
Appear_Chxx 219 7.55 0.66 0.87 à & ^B filter 
App. Background 102 3.52 0.52 0.40 do B filter 
SCR Debuglog 131 017 0.15 dod B filter 
SCR ViewController 1.86 0.17 0.21 A od B filter 
App. Mem Warn 1 0.03 0.03 0.00 d h B filter 
SCR ViewController Child 7 0.24 0.07 0.03 à & ^H filter 
SCR InteractiveNotification 19 0.66 0.07 0.08 d o B filter 
FV. Ph Strong 0 0.00 0.00 0.00 hae B filter 
FV_Ph Weak 0 0.00 0.00 0.00 do [5] filter 








1-6: 事件 一 一 更 重要 的 报告 


ur 1-7 中 的 崩溃 报告 


日 志 。 


， 你 会 注意 














到 其 中 的 download 链接 ， 这 个 链接 可 以 下 载 月 涡 


点 击 链 接 ， 下 载 日 志 。 看 着 很 熟悉 是 中? 








Full Stack Trace 


Full Stack Trace: 
CoreFoundation 
libobjc.A.dylib 
CoreFoundation 
vCircle 

UIKit 

UIKit 

UIKit 

UIKit 

UIKit 

UIKit 

10 UIKit 

11 UIKit 

12 UIKit 

13 CoreFoundation 
14 CoreFoundation 
15 CoreFoundation 
16 CoreFoundation 
17 CoreFoundation 


COIR PWNHO 


18 GraphicsServices 


19 UIKit 
20 vCircle 
21 libdyld.dylib 


Crash Report Data: 





Ox2fa3cecb 
Ox3ald7ce7 
Ox2fa3ce0d 
0x0001833b 
0x322a36a7 
0x322a3643 
0x322a3613 
0x3228ed5b 
0x322a305b 
0x322a2d2d 
0x3229dc87 
0x32272e55 
0x32271521 
Ox2fa07 fat 
Ox2£a07477 
Ox2£a05c67 
0x2£970729 
0x2£97050b 
0x348df6d3 
0x322d1871 
0x00018195 
Ox3a6d5ab7 


Desym File: ,4 upload a new desym file 


<redacted> + 130 

_objc_exception_throw + 38 

-[NSException initWithCoder:] + 0 
-[HPVFirstViewController crashButtonWasClicked:] + 122 
-[UIApplication sendAction:to:from:forEvent:] + 90 
-[UIApplication sendAction:toTarget:fromSender:forEvent:] + 38 
-[UIControl sendAction:to:forEvent:] + 46 
-[UIControl _sendActionsForEvents:withEvent:] + 374 
-[UIControl touchesEnded:withEvent:] + 594 
-[UIWindow _sendTouchesForEvent:] + 528 

-[UIWindow sendEvent:] + 758 

-[UIApplication sendEvent:] + 196 

<redacted> + 7120 

<redacted> + 14 

<redacted> + 206 

<redacted> + 630 

_CFRunLoopRunSpecific + 524 

.CFRunLoopRunInMode + 106 

_GSEventRunModal + 138 

“UIApplicationMain + 1136 

_main + 116 

<redacted> + 2 











1-7; 


崩溃 报告 一 一 最 重要 的 报告 








在 iTunes Connect 中 查看 骨 溃 报告 


Apple 提供 了 下 载 崩 演 报 告 的 服务 。 你 可 以 利用 该 服务 下 载 TestFlight 或 App Store 所 
发 布 的 应 用 的 最 近 版 本 ， 也 可 以 构建 相应 的 前 溃 报 告 。 理 论 上 来 说 ， 有 了 这 个 服务 就 
不 再 需要 第 三 方 的 前 溃 报 告 工具 了 。 











AAR YHA, RAE P ELE BARA eal red 否则 崩溃 日 志 不 会 发 
送 至 Apple, TestFlight 用 户 自动 同意 了 分 享 崩 FA 但 产品 应 用 (通过 App Store 
分 发 的 应 用 ) 必须 由 用 户 开启 分 享 。 


要 想 实现 这 一 点 ， 用 户 需 要 进入 设置 应 用 ， 打开 “隐私 一 诊断 与 用 量 ”(Privacy 一 
Diagnostics & Usage) ， 然 后 选择 “自动 发 送 ”(Automatically Send) 选项 (UA 1-8). 





eeocc 9 729588 ) 


< Privacy Diagnostics & Usage 


Automatically Send 


Don’t Send v 
Help Apple improve its products and services by automatically 


sending daily diagnostic and usage data. Diagnostic data may 
include location information. About Diagnostics & Privacy... 


Diagnostic & Usage Data 


Diagnostic & usage data will not be sent to Apple. 











图 1-8: 在 设备 中 设置 发 送 崩 演 报 告 

这 里 存在 两 个 问题 。 首 先 ， 用 户 不 能 在 你 的 应 用 内 设置 这 个 选项 ， 他 们 需要 进入 设置 
应 用 并 进入 特定 的 设置 项 。 第 二 点 更 为 重要 ， 这 项 设置 会 对 所 有 的 应 用 生效 : 用 户 无 
法 只 为 特定 的 应 用 发 送 崩 演 报 告 。 

使 用 第 三 方 的 崩 演 报告 工具 可 以 确保 你 能 够 控制 整体 体验 和 用 户 设置 ， 以 便 向 服务 器 
AE RIE 








1.44 Hi 
Aart ze, "DAUDUHOTOAUSNGHAEIITAG. 


日 志和 埋 点 之 间 存在 着 细微 的 差别 。 埋 点 可 以 看 作 日 志 的 子 集 。 被 埋 点 的 任何 数据 都 应 该 
记录 在 日 志 中 。 
埋 点 承担 了 为 聚合 分 析 发 布 关键 性 能 数据 的 职责 ， 日 志 则 提供 了 用 于 在 不 同 级 别 跟踪 应 用 


的 细节 信息 ， 比 如 debug, Verbose, info, warning 和 Error。 日 志 的 记录 会 贯穿 应 用 的 整 
个 生命 周期 ， 而 埋 点 只 应 该 用 在 开发 的 特定 阶段 。 


3H 
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埋 点 数据 会 发 送 到 服务 器 ， 日 志 是 记录 在 设备 本 地 。 
就 日 志 而 言 ， 我 们 可 以 通过 CocoaPods 引入 CocoaLumberjack 来 使 用 。 


例 1-5 展示 了 添加 到 Podfile 的 一 行 代码 ， 以 便 引 入 库 。 完 成 相关 改动 后 ， 运 行 pod update 
以 更 新 Xode 的 工作 空间 。 

















例 1-5 为 CocoaLumberjack 配置 Podfile 


pod 'CocoaLumberjack', '~> 2.0' 


CocoaLumber jack z— Ap HEPEIR SERES, ZB TAIA EAA ERA, AERA 
可 以 向 不 同 的 目标 发 送信 息 。 例 如 ， EH osse 可 以 向 Appie System Los (ASL, 
NSLog 方法 的 默认 位 置 ) 记录 日 志 。 类 似 地 ， 使 用 DDFileLogger 可 以 向 文件 记录 日 志 。 
以 在 应 用 运行 期 间 配 置 记录 器 
DDLog<Level> 宏 指 令 可 以 用 于 记录 某 个 特定 层级 的 日 志 。 层 级 越 高 ， 信 息 越 重要 。 最 高 级 
别 是 Error， 最 低级 别 是 Verbose。 实 际 记录 消息 的 最 低层 级 可 以 配置 在 每 个 文件 层级 、 每 
个 Xcode 配置 层级 、 每 个 日 志 器 层级 或 全 局 。 
以 下 的 宏 指 令 可 供 使 用 。 
e DDLogError 

表示 不 可 恢复 的 错误 。 
e DDLogWarn 

表示 可 恢复 的 错误 。 
e DDLogInfo 

表示 非 错误 的 信息 。 
e DDLogDebug 

表示 数据 主要 用 于 调试 。 
e DDLogVerbose 


几乎 提供 了 所 有 的 细节 ， 主 要 用 于 跟 踩 执行 过 程 中 的 控制 疲 。 


这 些 宏 指 令 有 着 与 NSLog 相同 的 签名 。 这 意味 着 你 可 以 直接 用 适合 的 DDLog<Level> 调用 来 
取代 NSLog。 


例 1-6 展示 了 配置 和 使 用 这 个 库 的 代表 性 代码 。 
例 1-6 配置 和 使 用 CocoaLumberjack 


// 设 置 
-(void)setupLogger { @ 








Hif _DEBUG 
[DDLog addLogger:[DDASLLogger sharedInstance]]; @ 
#endif 


DDFileLogger fileLogger = [[DDFileLogger alloc] init]; 9 
fileLogger.rollingFrequency = 60 * 60 * 24; 





fileLogger.logFileManager.maximumNumberOfLogFiles = 7; 


[DDLog addLogger:fileLogger]; @ 


} 
// 在 一 些 文件 中 使 用 记录 器 
iif _DEBUG @ 


static const DDLogLevel ddLogLevel = DDLogLevelVerbose; 
#elsif MY INTERNAL RELEASE 

static const DDLogLevel ddLogLevel - DDLogLevelDebug; 
#else 

static const DDLogLevel ddLogLevel = DDLogLevelWarn; 
#end 


-(void)someMethod { 
DDVerbose(@"someMethod has started execution"); @ 


//..- 

DDError(@"Ouch! Error state. Don't know what to do"); 
//..- 

DDVerbose(@"someMethod has reached its end state"); 


} 


Q 最 有 可 能 在 application: didFinishLaunchingWithOptions: 调用 这 个 方法 。 

四 当 连 接 到 Xcode 时 ， 只 在 调试 模式 下 将 日 志 记 录 到 ASL。 你 不 会 希望 这 类 日 志 在 产 品 
环境 的 设备 中 记录 下 来 。 

@ 文件 日 志 记 录 器 ， 配 置 为 每 24 小 时 (rollingFrequence) 创建 一 个 新 文件 ， 同 时 最 多 
允许 创建 7 个 文件 (maximumNumberOfLogFiles), 

O 注册 日 志 记 录 器 。 

Q 将 日 志 的 级 别 (ddLogLevel) 设置 为 合适 的 值 。 这 里 我 们 可 以 这 样 设置 : 开发 阶段 输出 
最 多 的 细节 ， 内 部 测试 阶段 (MY_INTERNAL_RELEASE 是 一 个 自 定 义 的 标记 ) 输出 少量 细 
节 信 息 (debug level) ; 面向 终端 用 户 的 分 发 包 只 输出 错误 信息 。 

Q 记录 一 些 信息 。 在 DDLogLevelverbose 级 别 中 ， 所 有 信息 都 会 被 记录 ;在 DDLogLevelWarn 
级 别 中 ， 只 有 错误 信息 会 被 记录 。 








建议 在 应 用 委托 的 application:didFinishLaunchingWithOptions: 回调 中 
调用 此 方法 。 








1.5 小结 
我 们 在 本 章 了 解 了 影响 应 用 性 能 的 因素 。 性 能 不 仅 涉 及 用 户 体验 ， 更 关系 到 应 用 能 否 高 效 


运行 。 
我 们 查看 了 一 些 影响 应 用 性 能 的 关键 属性 。 在 包含 测量 和 追踪 的 各 项 指标 中 ， 这 些 属 性 成 
为 了 性 能 的 关键 指示 器 。 















































我 们 讨论 了 性 能 分 析 的 概念 ， 并 宽泛 地 介绍 了 两 类 分 析 技 术 : 采样 与 埋 点 。 还 介绍 了 一 些 
代码 改动 ， 以 便 对 应 用 进行 埋 点 。 然 后 我 们 通过 使 用 被 埋 点 的 应 用 来 触发 事件 。 
最 后 ， 我 们 在 类 中 添加 一 些 样板 代码 以 帮助 进行 埋 点 和 记录 日 志 。 





第 二 部 分 的 音节 主要 关注 定义 性 能 的 每 个 属性 。 每 章 都 从 定义 和 评审 属性 开始 ， 然 后 讨论 
一 些 潜 在 的 问题 ， 并 用 实际 的 代码 来 解决 这 些 问题 





第 二 部 分 


核心 优化 





我 们 即将 探讨 最 核心 的 优化 ， 以 便 能 够 使 用 Objective-C 编写 高 效 的 应 用 。 这 些 优化 构成 了 
每 个 应 用 的 基础 ， 它 们 无 处 不 在 。 这 些 优化 和 具体 选用 的 API 无 关 ， 和 在 应 用 的 哪 一 层 实 
现 无 关 ， 黄 至 和 应 用 的 目标 也 没什么 关系 ， 因 为 它们 在 整个 应 用 领域 都 适用 。 

我 们 要 讨论 的 优化 包括 以 下 方面 : 

。 内 存 管理 

。 能 


。 并 发 编程 





第 2 章 


内 存 管理 





iPhone 和 iPad 设备 的 内 存 资源 非常 有 限 。 如 果 某 个 应 用 的 内 存 使 用 量 超过 了 单个 进程 的 上 
BR, 那么 它 就 会 被 操作 系统 终止 使 用 。 ' 正 是 由 于 这 个 原因 ,成功 的 内 存 管理 在 OS 应 用 的 
实现 过 程 中 扮演 着 核心 的 角色 。 
苹果 公司 在 2011 年 的 全 球 开发 者 大 会 上 指出 ，90% 的 应 用 崩溃 与 内 存 管理 有 关 。 其 中 最 
主要 的 原因 是 错误 的 内 存 访 问 和 保留 环 所 引起 的 内 存 泄漏 。 

与 (基于 垃圾 回收 的 ) Java 运行 时 不 同 ，Objective-C 和 Swift 的 iOS 运行 时 使 用 引用 计数 。 
使 用 引用 计数 的 负面 影响 在 于 ， 如 果 开 发 人 员 不 够 小 心 ， 那 么 可 能 会 出 现 重复 的 内 存 释 放 
和 循环 引用 的 情况 。 

因此 ， 理 解 OS 的 内 存 管理 是 十 分 重要 的 。 

我 们 将 在 本 章 学 习 以 下 知识 点 : 

。 内 存 消耗 (例如 ， 应 用 如 何 消耗 内 存 ) 

。 内 存 管理 模型 (例如 ，iOS 运行 时 如 何 管理 内 存 ) 

。 语言 架构 我 们 将 介绍 Objective-C 的 架构 及 一 些 实用 特性 

。 在 不 影响 用 户 体验 的 前 提 下 ， 采 用 减少 内 存 使 用 的 最 佳 实践 


2.1 内 存 消耗 


内 存 消耗 指 的 是 应 用 消耗 的 RAM., 






























































iE 1: iOS Developer Library, "Technical Note TN2151: Understanding and Analyzing iOS Application CrashReports 
(https://developer.apple.com/library/ios/technotes/tn2 15 1/_index.html#//apple_ref/doc/uid/DTS40008 184-CH1- 
UNDERSTANDING LOW MEMORY REPORTS). 
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iOS 的 虚拟 内 存 模型 并 不 包含 交换 内 存 ， 与 桌面 应 用 不 同 ， 这 意味 着 磁盘 不 会 被 用 来 分 页 
内 存 。 最 终 的 结果 是 应 用 只 能 使 用 有 限 的 RAM。 这 些 RAM 的 使 用 者 不 仅 包括 在 前 台 运 行 
的 应 用 ， 还 包括 操作 系统 服务 ， 甚 至 还 包括 其 他 应 用 所 执行 的 后 台 任 务 。 


应 用 中 的 内 存 消 耗 分 为 两 部 分 : 栈 大 小 和 堆 大 小 。 接 下 来 的 两 市 将 逐一 介绍 这 两 部 分 。 


2.1.1. 栈 大 小 

应 用 中 新 创建 的 每 个 线程 都 有 专用 的 栈 空间 ， 该 空间 由 保留 的 内 存 和 初始 提交 的 内 存 组 

成 。 栈 可 以 在 线程 存在 期 间 自由 使 用 。 线 程 的 最 大 栈 空间 很 小 ， 这 就 决定 了 以 下 的 限制 。 

。 可 被 递归 调用 的 最 大 方法 数 
每 个 方法 都 有 其 自己 的 栈 帧 ， 并 会 消耗 整体 的 栈 空间 。 例 如 ， 如 例 2-1 所 示 ， 如 果 你 调 

用 main, JBA main 将 调用 methodi, my methodi 又 将 调用 methodz2， 这 就 存在 三 个 栈 帧 

了 ， 且 每 个 栈 帧 都 会 消耗 一 定 字 节 的 内 存 。 图 2-1 展示 了 线程 栈 随时 间 的 变化 。 


例 2-1 调用 树 


main() { 
method1(); 













































































methodi() { 
method2(); 
} 


。 一 个 方法 中 最 多 可 以 使 用 的 变量 个 数 
所 有 的 变量 都 会 载 和 方法 的 栈 帧 中 ， 并 消耗 一 定 的 栈 空间 。 


。 视图 层级 中 可 以 内 入 的 最 大 视图 深度 
泻 染 复合 视图 将 在 整个 视图 层级 树 中 递归 地 调用 LayoutSubViews 和 drawRect 方法 。 如 


果 层 级 过 深 ， 可 能 会 导致 栈 溢出 。 
- 


method1 method1 method1 





























































栈 上 消耗 的 内 存 








执行 的 时 间 轴 





B21: 包含 每 个 方法 的 栈 框架 的 栈 








21.2 HEX 


每 个 进程 的 所 有 线程 共享 同一 个 堆 。 一 个 应 用 可 以 使 用 的 堆 大 小 通常 远 远 小 于 设 









































的 


RAM 值 。 例 如 ，iPhone 5S 拥有 大 约 1GB 的 RAM， 但 分 配给 一 个 应 用 的 堆 大 小 最 多 不 到 











512MB 。 应 用 并 不 能 控制 分 配给 它 的 堆 。 只 有 操作 系统 才能 管理 堆 。” 


使 用 Nsstring、 载 和 图片 、 创 建 或 使 用 JSON/XML 数据 、 使 用 视图 等 都 会 消耗 大 量 的 堆 
内 存 。 如 果 你 的 应 用 大 量 使 用 图 片 (与 Flickr 和 Instagram 应 用 类 似 )， 那 么 你 需要 格外 关 









































注 平 均值 和 峰值 内 存 使 用 的 最 小 化 。 
图 2-2 展示 了 可 能 出 现在 一 个 应 用 某 个 时 刻 的 一 个 典型 扒 。 





BE 
































UITableViewDataSource 的 tableView:cellForRowAtIndex: 方法 。 


在 图 2-2 中， 由 main 方 法 启动 的 主线 程 创 建 了 M nly 我 们 假设 某 个 时 间 点 
的 窗 体 包含 了 一 个 UITableView， 当 必须 演 染 表格 中 的 一 行 时 ，UITableVview 调 用 了 


通过 名 为 photos 的 NSArray 属性 ， 数 据 源 引用 了 全 部 的 照片 。 如 果 人 处 理 不 够 谨慎 ， 这 个 
数组 将 会 非常 大 ， 从 而 导致 很 高 的 峰值 内 存 使 用 。 解 决 方案 之 一 是 在 数组 中 存储 固定 数量 
的 图 片 ， 并 在 用 户 滚动 视图 时 换 入 或 换 出 图 片 。 这 个 固定 的 数值 将 决定 此 应 用 的 平均 内 存 












































使 用 。 





线程 栈 








tableView: ; 
cellForRow uA 
Atindex: aoue 


















. creationDate 


Jan 1, 2010 











2-2; fg UITableViewDataSource 中 展示 HPPhoto 模型 使 用 情况 的 堆 


数组 中 的 每 一 项 都 是 HPPhoto 类 型 ， 代 表 了 一 张 照片 。HPPhoto 储存 了 与 对 象 有 关 的 数据 ， 
如 照片 的 尺寸 、 创 建 日 期 、 拥 有 者 信息 、 标 签 、 与 照片 关联 的 网 络 URL (图 中 没有 展示 )、 














对 本 地 缓存 的 引用 (图 中 没有 展示 )， 等 等 。 














注 2: Stack Overflow, “iOS Equivalent to Increasing Heap Size” (http://stackoverflow.com/a/25369670). 
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与 通过 类 创建 的 对 象 相 关 的 所 有 数据 都 存放 在 堆 中 。 

类 可 能 包含 属性 或 值 类 型 的 实例 变量 (iVars), Aint, char 或 struct。 但 因为 对 象 是 在 
堆 内 创建 的 ， 所 以 它们 只 消耗 堆 内 存 。 

当 对 象 被 创建 并 被 赋值 时 ， 数 据 可 能 会 从 栈 复制 到 堆 。 类 似 地 ， 当 值 仅 在 方法 内 部 使 用 
时 ， 它 们 也 可 能 会 被 从 堆 复制 到 栈 。 这 可 能 是 个 代价 昂贵 的 操作 。 例 2-2 重点 展示 了 从 栈 
复制 到 堆 以 及 从 堆 复制 到 栈 的 情况 。 

Gil 2-2 HEGRE 


@interface AClass @ 











@property (nonatomic, assign) NSInteger anInteger; @ 
@property (nonatomic, copy) NSString *aString; © 


Qend 











// 一 些 其 他 的 类 
-(AClass *) createAClassWithInteger:(NSInteger)i 
string:(NSString *)s ( @ 





AClass *result - [AClass new]; 
result.anInteger = i; © 
result.aString = s; © 


j 


-(void) someMethod:(NSArray *)items { @ 
NSInteger total - 0; 
NSMutableString *finalValue - [NSMutableString string]; 


for(AClass *obj in items) { 
total += obj.anInteger; @ 
[finalValue appendString:obj.aString]; © 
} 
} 


@ 类 Aclass 包含 两 个 属性 。 

@ anInteger 为 NSInteger 类 型 ， 通 过 传 值 方式 进行 传递 。 

© astring 7j NsString * 类 型 ， 通 过 引用 传递 。 

@ createAClassWithInteger:string: 方法 (存在 于 其 他 某 个 不 相关 的 类 ) 初始 化 了 Aclass, 
这 个 方法 拥有 创建 对 象 时 需要 的 值 。 

© i 的 值 在 栈 上 。 但 赋值 给 属性 时 ， 它 必须 被 复制 到 堆 中 ， 因 为 那 是 存储 result 的 地 方 。 

© 虽然 NSString * 通过 引用 传递 ， 但 这 个 属性 被 标记 为 copy。 这 意味 着 它 的 值 必须 被 复 
制 或 克隆 ， 这 取决 于 [-NSCopying copyWithzone:] 方法 的 实现 。 

@ someMethod: 方法 对 AClass 对 象 的 数组 进行 操作 。 

© 使 用 anInteger 时 ， 它 的 值 必 须 先 复制 到 栈 然 后 才能 进行 进一步 的 处 理 。 在 本 示例 中 ， 
它 的 值 加 到 total, 

© astring 在 使 用 时 通过 引用 传递 。 在 本 示例 中 ，appendString: 使 用 了 aString 对 象 的 引用 。 















































保持 应 用 的 内 存 需求 总 是 处 于 RAM 的 较 低 占 比 是 一 个 非常 好 的 主意 。 虽 然 
没有 强制 规定 ， 但 强烈 建议 使 用 量 不 要 超过 80%~85%， 要 给 操作 系统 的 核 
心服 务 留 下 足够 多 的 内 存 。 


不 要 忽视 didReceiveMemoryWarning 信号 。 





2.2 内存 管理 模型 


我 们 将 在 本 节 中 学 习 iOS 的 运行 时 是 如 何 管理 内 存 的 ， 以 及 它 对 代码 的 影响 。 


内 存 管理 模型 基于 桂 有 关系 的 概念 。 如 果 一 个 对 象 正 处 于 被 持 有 状态 ， 那 它 占 用 的 内 存 就 
不 能 被 回收 。 











[0 果 这 个 对 象 从 方法 
返回 , 则 调用 者 声称 建立 了 持 有 关系 。 这 个 值 可 以 赋值 ”给 其 他 变量 , 对 应 的 变量 同样 会 声 
称 建立 了 持 有 关系 。 
一 旦 与 某 个 对 象 相 关 的 任务 全 部 完成 ， 那 么 就 是 放弃 了 持 有 关系 。 这 一 过 程 没 有 转移 持 有 
关系 ， 而 是 分 别 增加 或 减少 了 持 有 者 的 数量 。 当 持 有 者 的 数量 降 为 零 时 ， 对 象 会 被 释放 
(https://developer.apple.com/library/mac/documentation/Cocoa/reference/Foundation/Classes/ 
nsobject_Class/Reference/Reference.html#//apple_ref/occ/instm/NSObject/dealloc), ， 相 关 的 内 
会 被 回收 。 
这 种 持 有 关系 计数 通常 被 正式 称 为 引用 计数 。 当 你 亲自 管理 时 ， 它 被 称 为 手动 引用 计数 
(manual reference counting，MRC)。 虽 然 现 在 已 经 十 分 罕见 ， 但 MRC 对 理解 问题 很 有 耕 
助 。 现 如 今 的 应 用 大 都 使 用 自动 引用 计数 (automatic reference counting ，ARC) ， 我 们 将 
在 2.5 节 中 对 其 进行 讨论 。 


例 2-3 演示 了 基于 引用 计数 的 手动 内 存 管理 的 基本 结构 。 
例 2-3 通过 手动 内 存 管理 进行 引用 计数 


NSString *message = Q"Objective-C is a verbose yet awesome language"; Qj 
NSString *messageRetained = [message retain]; @ 

[messageRetained release]; © 

[message release]; @ 

NSLog(@"Value of message: %@", message); @ 


@ 创建 对 象 、message 建立 了 持 有 关系 ， 引 用 计数 为 1。 

@ messageRetained 建立 了 持 有 关系 ， 引 用 计数 增加 为 2。 

© nessageRetained 放弃 了 持 有 关系 ， 引 用 计数 降 为 1。 

O nessage 放弃 了 持 有 关系 ， 引 用 计数 降 为 0。 

© 严格 来 讲 ， 此 时 message 的 值 是 未 定义 的 。 你 仍然 能 像 之 前 那样 得 到 相同 的 值 ， 因 为 它 
对 应 的 内 存 还 没有 被 回收 或 重 置 。 


例 2-4 演示 了 方法 是 如 何 对 引用 计数 产生 影响 的 。 







































































注 3: 术语 “赋值 ”在 此 处 使 用 的 并 不 精确 ， 我 们 将 在 后 文 做 详细 阐述 。 
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例 2-4 方法 中 的 引用 计数 
// 一 个 Person 类 的 部 分 
-(NSString *) address { 

NSString *result = [[NSString alloc] 
initwWithFormat:@"%@\n%@\n%e, %0", 
self.linei, self.line2, self.city, self.sta 

return result; 


-(void) showPerson:(Person *) p { 
NSString *paddress = [p address]; @ 
NSLog(@"Person's Address: %@", paddress); 


[paddress release]; © 
} 


Q 首次 创建 对 象 ，result 指向 内 存 的 引用 计数 为 1。 





@ 通过 paddress (指向 result) 指向 的 内 存 的 引用 计数 仍然 是 1。 


te]; @ 


showPerson: 方法 通过 


address 按钮 创建 了 对 象 ， 是 对 象 的 持 有 者 。 对 象 不 应 该 被 再 次 持 有 (retain), 


O 放弃 持 有 关系 ;引用 计数 降 为 0。 








在 例 2-4 中 ，showPerson: 并 不 知道 address 方法 是 创建 了 一 个 新 的 对 象 还 是 重用 了 旧 的 对 
象 。 但 它 知 道 引 用 计数 加 1 后， 对 象 会 被 返回 。 因 此 ， 这 里 没有 继续 持 有 address, — H, 
完成 任务 ， 它 将 释放 对 象 。 如 果 对 象 的 引用 计数 是 1， 那 么 它 将 变 成 0， 并且 被 回收 。 














人 苹果 公司 和 LLVM 的 官方 文档 更 喜欢 使 用 术语 持 有 关系 。 本 
计数 这 两 个 术语 。 


2.3 BEN 


自动 释放 对 象 让 你 能 够 放弃 对 一 个 对 象 的 持 有 关系 ， 但 延 后 对 它 的 销毁 。 当 在 方法 中 创建 














将 交替 使 有 











日 持 有 关系 和 引用 





一 个 对 象 并 需要 将 其 返回 时 ， 自 动 释放 就 显得 非常 有 用 。 自 动 释放 可 以 帮助 在 MRC 中 管 


























n 








里 对 象 的 生命 周期 。 


以 严格 的 Objective-C 命名 规范 为 标准 ， 在 例 2-4 中 ， 没 什么 能 表示 address 方法 持 有 了 返 
回 的 字符 串 。 因 此 ， 方法 的 调用 者 showPerson: 也 不 应 该 释放 返回 的 字符 串 ， 这 可 能 会 导 





致 发生 内 存 泄漏 。 加 入 [paddress release] 这 行 代 码 的 目的 是 为 了 指明 这 种 情况 。 


那么 ， 正 确 使 用 address 方法 的 代码 是 什么 样 的 呢 ? 
以 下 是 两 种 可 能 的 解决 方案 。 


。 不 要 使 用 alloc 或 相关 的 方法 。 
。 对 返回 的 对 象 使 用 延 时 释放 。 





使 用 NSString 时 ， 第 一 个 修复 版 本 很 容易 实现 。 更 新 后 的 代码 见 例 2-5。 





例 2-5 方法 中 引用 计数 的 修复 代码 
-(NSString *) address { 
NSString *result - [NSString 
stringWithFormat:@"%@\n%@\n%a, %0", 
self.linei, self.line2, self.city, self.state]; @ 
return result; 


} 


-(void) showPerson:(Person *) p { 
NSString *paddress = [p address]; 


NSLog(@"Person's Address: %@", paddress); 
} 


Q 不 要 使 用 alloc 方法 。 

@ 由 于 showPerson: 方法 没有 创建 实体 对 象 ， 因 此 不 要 在 showPerson: 方法 中 使 用 release 
方法 。 

然而 ， 这 种 修复 方法 在 没有 使 用 NSString 的 情况 下 并 不 容易 实现 ， 因 为 通常 很 难 找到 能 

够 满足 需要 的 适合 方法 。 例 如 ， 当 使 用 第 三 方 类 库 或 者 某 个 类 有 多 个 用 于 创建 对 象 的 方法 

时 ， 到 底 是 哪个 方法 保持 了 持 有 关系 并 不 明确 。 

延迟 销毁 大 显 神 威 的 机 会 来 了 。 

NSObject 协议 定义 了 可 被 用 于 延迟 释放 的 autorelease 消息 。 可 在 从 方法 中 返回 对 象 时 使 

HE. 

[8] 2-6 显示 了 使 用 autorelease 的 更 新 代码 。 

例 2-6 使 用 autorelease 的 引用 计数 


-(NSString *) address 
































NSString *result - [[[NSString alloc] 
initWithFormat:@"%@\n%@\n%e, %0", 
self.line1, self.line2, self.city, self.state] 
autorelease]; 
return result; 


} 
可 以 用 以 下 规则 来 分 析 代码 。 


(D) 持 有 的 对 象 《在 上 述 示例 中 是 NsString) 是 alloc 方法 返回 的 。 

(2) 确保 没有 内 存 泄漏 ， 你 必须 在 失去 引用 之 前 放弃 持 有 关系 。 

(3) 但 是 ， 如 有 果 使 用 了 retease， 那 么 对 象 的 释放 将 发 生 在 返回 之 前 ， 因 而 方法 将 返回 一 个 
无 效 的 引用 。 

(4) autorelease 表明 你 想 要 放弃 持 有 关系 ， 同 时 允许 方法 的 调用 者 在 对 象 被 释放 之 前 使 用 
对 象 。 
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当 创 建 一 个 对 象 并 将 其 从 非 alloc 方法 返回 时 ， 应 使 用 autorelease, ixfÉ 
可 以 确保 对 象 将 被 释放 ， 并 尽量 在 调用 方法 执行 完成 时 立即 释放 。 








2.4 目 动 释放 池 块 

自动 释放 池 块 是 允许 你 放弃 对 一 个 对 象 的 持 有 关系 、 但 可 避免 它 立 即 被 回收 的 一 个 工具 。 
当 从 方法 返回 对 象 时 ， 这 种 功能 非常 有 用 。 

它 还 能 确保 在 块 内 创建 的 对 象 会 在 块 完成 时 被 回收 。 这 在 创建 了 多 个 对 象 的 场景 中 非常 有 
用 。 本 地 的 块 可 以 用 来 尽早 地 释放 其 中 的 对 象 ， 从 而 使 内 存 用 量 保 持 在 较 低 的 水 平 。 

自动 释放 池 块 用 Gautoreleasepool 表示 。 

打开 示例 工程 的 main.m 文件 ， 你 会 发 现 例 2-7 中 的 代码 。 





























例 2-7 main.m 中 的 @autoreleasepool 块 


int main(int argc, char * argv[]) { 
@autoreleasepool { 
return UIApplicationMain(argc, argv, nil, 
NSStringFromClass([HPAppDelegate class])); 
} 
} 


块 中 收 到 过 autorelease 消息 的 所 有 对 象 都 会 在 autoreleasepool 块 结束 时 收 到 release TH 
息 。 更 加 重要 的 是 ， 每 个 autorelease 调用 都 会 发 送 一 个 release 消息 。 这 意味 着 如 果 一 
个 对 象 收 到 了 不 止 一 次 的 autorelease 消息 ， 那 它 也 会 多 次 收 到 release 消息 。 这 一 点 很 
棒 ， 因 为 这 能 保证 对 象 的 引用 计数 下 降 到 使 用 autoreleasepool 块 之 前 的 值 。 如 果 计 数 为 
0， 则 对 象 将 被 回收 ， 从 而 保持 较 低 的 内 存 使 用 率 。 


看 了 main 方法 的 代码 后 ， 你 会 发 现 整个 应 用 都 在 一 个 autoreleasepool 块 中 ， 这 意味 着 所 
有 的 autorelease 对 象 最 后 都 会 被 回收 ， 不 会 导致 内 存 泄漏 。 


与 其 他 的 代码 块 一 样 ，autoreLeasepoot KALA MRE, Ani 2-8 所 示 。 









































例 2-8 RÆKJ autoreleasepool Le 


@autoreleasepool { 
// 一 些 代码 
@autoreleasepool { 


// Rm 
} 
E Ay We — 4S Jj REA IB — AU a HA, TE Ta] — 2; EAN E H BA AY 
autoreleasepool 块 并 不 常见 。 但 是 ， 被 调用 的 方法 也 可 能 拥有 自己 的 autoreleasepool 
块 ， 以 提前 执行 对 象 的 回收 。 
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自动 释放 池 块 无 处 不 在 


Cocoa 框架 希望 代码 能 在 autoreleasepool 块 内 执行 ， 否则 autorelease 对 象 将 无 法 被 
释放 ， 从 而 导致 应 用 发 生 内 存 泄 漏 。 


AppKit 和 UIKit 框架 将 事件 - 循环 的 选 代 放 入 了 autoreleasepool 块 中 。 因 此 ， 通 常 
不 需要 你 自己 再 创建 autoreleasepool 块 了 。 





但 在 一 些 特定 情况 下 ， 你 很 可 能 想 创 建 自 己 的 autereleasepool 块 ， 例 如 以 下 这 些 情况 。 





当 你 有 一 个 创建 了 很 多 临时 对 象 的 循环 时 

在 循环 中 使 用 autoreleasepool 块 可 以 为 每 个 迭代 释放 内 存 。 虽 然 迭 代 前 后 最 终 的 内 存 
使 用 相同 ， 但 你 的 应 用 的 最 大 内 存 需求 可 以 大 大 降低 。 

例 2-9 提供 了 一 些 例子 ， 甚 中 包括 使 用 autoreleasepool 时 编写 的 好 与 不 好 的 代码 实现 。 
当 你 创建 一 个 线程 时 

每 个 线程 都 将 有 它 自 己 的 autoreleasepool 块 栈 。 主 线程 用 自己 的 autoreleasepool 局 
动 ， 因 为 它 来 自 统一 生成 的 代码 。 然 而 ， 对 于 任何 自 定 义 的 线程 ， 你 必须 创建 自己 的 


autoreleasepool, 


你 可 以 通过 例 2-10 查看 示例 代码 。 








例 2-9 循环 中 的 自动 释放 凶 块 


// 不 良 代码 @ 
{ 





@autoreleasepool { 
NSUInteger *userCount = userDatabase.userCount; 


for(NSUInteger *i = 0; i < userCount; i++) { 
Person *p = [userDatabase userAtIndex:i]; 


NSString *fname = p.fname; 
if(fname == nil) { 
fname = [self askUserForFirstName] ; 


j 


NSString *lname - p.lname; 
if(lname == nil) { 
lname = [self askUserForLastName]; 
} 
//... 


[userDatabase updateUser:p]; 


} 
// 好 的 代码 e 
{ 


@autoreleasepool { 
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NSUInteger *userCount = userDatabase.userCount; 
for(NSUInteger *i = 0; i « userCount; i++) { 


@autoreleasepool { 
Person *p = [userDatabase userAtIndex:i]; 


NSString *fname = p.fname; 
if(fname == nil) { 
fname = [self askUserForFirstName]; 


} 


NSString *lname = p.lname; 
if(lname == nil) { 
lname = [self askUserForLastName]; 
} 
sss 


[userDatabase updateUser:p]; 


j 


Q 这 段 代 码 很 糟糕 ， 因 为 只 有 一 个 autoreteasepooL， 而 且 内 存 清理 工作 要 在 所 有 的 循环 
迭代 完成 之 后 才能 进行 。 

O 这 个 示例 中 有 两 个 autoreleasepool， 内 层 的 autoreleasepool 确保 在 每 次 循环 迭 代 完 成 
后 清理 内 存 ， 从 而 导致 更 少 的 内 存 需 求 


例 2-10 自 定义 线程 中 的 自动 释放 池 块 
-(void)myThreadStart:(id)obj { 
@autoreleasepool { 








// 新 线程 的 代码 
} 
} 
// 其 他 地 方 
{ 
NSThread *myThread = [[NSThread alloc] initWithTarget:self 
selector :@seLector(myThreadStart: ) 
object:nil]; 
[myThread start]; 
} 


2.5 自动 引用 计数 
持续 跟踪 retain, release 和 autorelease 并 不 容易 。 要 想 找 出 是 谁 在 什么 时 间 和 地 点 向 谁 
发 送 了 这 些 消息 就 更 难 了 。 


aid 司 在 2011 年 的 全 球 开 发 者 大 会 上 介绍 了 解决 这 一 问题 的 方案 一 一 ARC。iOS 应 用 的 
兴 语 言 Swift 同样 也 在 使 用 ARC。 与 Objective-C 不 同 的 是 ，Swift 不 支持 MRC。 























ARC 是 一 种 编译 器 特性 “。 它 评估 了 对 象 在 代码 中 的 生命 周期 ,并 在 编译 时 自动 注入 适合 的 
内 存 管理 调用 。 编 译 器 还 会 生成 适合 的 dealloc 方法 。 这 意味 着 与 跟踪 内 存 使 用 (如 确保 
对 象 被 及 时 回收 了 ) 有 关 的 最 大 难题 被 解决 了 。 


图 2-3 演示 了 使 用 MRC 5 ARC 的 开发 时 间 对 比 。 由 于 代码 减少 ， 使 用 ARC 开发 的 进程 
会 大 大 加 快 。 











应 用 逻辑 
保留 /释放 逻辑 


应 用 逻辑 


开发 周期 


保留 /释放 逻辑 


应 用 逻辑 


保留 /释放 逻辑 


手动 引用 计数 | 






































& 2-3: ARC 减少 了 开发 时 间 ,， 减轻 了 负担 


你 需要 确保 在 Xcode 工程 设置 中 开启 了 ARC， 这 从 Xcode 5 开始 成 为 一 项 默认 设置 ( 见 
图 2-4), 











器 | > |Ñ Circe <a> 














[zu] General Capabilities Info Build Settings Build Phases Build Rules 
PROJECT Basic (All) | Levels | + (Qr automatic 9) 
+ APNG Evi aii a | mvwwrsa 
vCircle 
Bic Setting Ay vCircle 
TARGETS Link Frameworks Automatically Yes $ 
(“lvCircleTests Y Apple LLVM 5.1 - Language - Objective C 


Setting Circle 


Objective-C Automatic Reference Counting Yes $ 


* Apple LLVM 5.1 - Warnings - All languages 
Setting Ae vCircle 








* = © 7 Uninitialized Variables Yes (Aggressive) + 

















图 2-4. ARC 在 Xcode 中 的 项 目 设 定 





iE 4: 你 可 以 从 LLVM 的 站 点 找到 自动 引用 计数 的 完整 规范 (http://clang.llvm.org/docs/AutomaticReference- 
Counting.html) E 
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没有 使 用 ARC 的 依赖 项 
现 阶 段 不 支持 ARC 或 没有 相关 解决 方案 的 依赖 项 目 已 经 非常 少 了 。 
但 如 果真 的 遇 到 了 这 样 的 依赖 项 目 ， 那 么 你 需要 对 一 个 或 多 个 文件 禁用 ARC, 


要 想 禁 用 ARC， 需 进入 Targets->Build Phases->Compile Sources， 选 择 必 须 禁 用 ARC 
的 文件 ， 然 后 添加 编译 器 标记 -fno-objc-arc， 如 图 2-5 所 示 。 





Ba |< > | B HighPerformance 
Oo General Capabilities Info Build Settings Build Phases Build Rules 
PROJECT + Q 
[à HignPertormance Y Compile Sources (50 items) x 
TARGETS Name Compiler Flags 


^ HighPerformance 
pz m HPChapter08_02ViewsTableViewController.m ...in ViewControllers 
(“JHighPerformanceTests 

m HPChapter03AllTableViewController.m ...in ViewC: 





E) actionRead 
C ghard Bead m HPChapter17Tools O2AllocationsViewController.m -fno-objc-ard 
TJ docümeripróuidar m HPChapter17Tools_04NetworkViewController.m .in View: 
E) documentProviderFile... m HPPhoto.m ...in Models 

m HPChapter09 DataSharingViewController.m n ViewControllers 

m HPMeasurableView.m n App/Views 

m HPChapter08_ChildViewController.m ...in ViewControllers 

m HPAppDelegate.m ...In HighPerformance 

m HPChapter17Tools_PonyDebuggerViewController.m ...in ViewControllers 

m HPChapter17Tools O3LeaksViewController.m ...in ViewControllers 

m NSltemProvider-HPExtension.m ...in App/Categories 


+- 6 


m HPConfiauratinnitemViewCantrollerm in shareRead 











2-5: 在 文件 级 禁用 ARC 


相同 的 选项 可 用 于 创建 混合 模式 的 类 ， 这 个 类 可 以 包含 基于 MRC 的 Category。 代 码 
写 在 Category 的 文件 中 ， 对 应 的 文件 可 以 禁用 ARC。 








ARC 的 规则 


ARC 强制 推行 了 编写 代码 时 需要 遵循 的 一 些 规 则 。 这 些 规 则 的 意图 是 提供 一 个 明确 的 内 存 
管理 模型 。 在 某 些 情况 下 ， 这 些 规则 的 目的 是 强制 实施 最 佳 实践 ， 在 其 他 情况 下 ， 它 们 可 
以 简化 代码 ,直接 的 必然 结果 就 是 开发 人 员 不 必 直 接 进 行内 存 管理 。” 这 些 规则 由 编译 器 强 
制 执行 ， 会 导致 编译 期 时 的 错误 而 不 是 运行 时 的 月 并。 以 下 就 是 编译 器 的 ARC 规则 。 


。 不 能 实现 或 调用 retain、release、autorelease 或 retainCount 方法 。 这 一 限制 不 仅 针 
对 对 象 ,对 选择 器 同样 有 效 。 因 此 ,[obj release] 或 @selector(retain) 是 编译 时 的 错误 。 
。 可 以 实现 dealloc 方法 ， 但 不 能 调用 它们 。 不 仅 不 能 调用 其 他 对 象 的 dealloc Wik, th 
不 能 调用 超 类 。[super dealloc] 是 编译 时 的 错误 。 
但 你 仍然 可 以 对 Core Foundation 类 型 的 对 象 调 用 CFRetain, CFRelease 等 相关 方法 。 


。 不 能 调用 NSALLocate0bject 和 NSDeallocateObject 方法 。 应 使 用 alloc 方法 创建 对 象 ， 
运行 时 负责 回收 对 象 。 







































































iE 5: iOS Developer Library, “Transitioning to ARC Release Notes" (http://apple.co/1KfifW3](http://apple.co/1KfifW3). 











。 不 能 在 C 语言 的 结构 体内 使 用 对 象 指 针 。 

。 不 能 在 id 类 型 和 void * 类 型 之 间 自 动 转 换 。 如 果 需 要 ， 那 么 你 必须 做 显示 转换 。 

。 不 能 使 用 NSAutoreleasePool， 要 替换 使 用 autoreleasepool H, 

。 不 能 使 用 NSzone 内 存 区 域 。 

。 属性 的 访问 器 名 称 不 能 以 new 开头 ,以 确保 与 MRC 的 互 操作 性 。 例 2-11 演示 了 这 一 点 。 

。 虽然 总 的 来 说 需要 避免 许多 事情 ， 但 仍然 可 以 混合 使 用 ARC 和 MRC 代码 (我 们 在 本 
市 前 面 讨 论 过 了 这 个 问题 )。 


例 2-11 开启 了 ARC 后 的 访问 器 名 称 


// 未 允许 
@property NSString * newTitle; 


LIR 
@property (getter=getNewTitle) NSString * newTitle; 


牢记 这 些 规 则 ， 我 们 可 以 更 新 例 2-5 中 的 代码 。 更 新 后 的 代码 见 例 2-12, 
例 2-12 开启 ARC 后 被 更 新 的 代码 


-(NSString *) address 
{ 
NSString *result = [[NSString alloc] initWithFormat:@"%@\n%@\n%@, %@", 
self.line1, self.line2, self.city, self.state]; @ 
return result; 






































-(void) showPerson:(Person *) p 


{ 
NSString *paddress = [p address]; 
NSLog(@"Person's Address: %@", paddress); 


} 


Q 此 处 无 需 调 用 autorelease。 你 不 能 在 result 对 象 上 调用 autorelease 或 retain 方法 。 
@ 你 不 能 再 对 paddress 调用 release 方法 。 


2.6 引用 类 型 


ARC 带 来 了 新 的 引用 类 型 : 弱 引 用 。 深 入 理解 这 些 引 用 类 型 对 内 存 管理 非常 重要 。 支 持 的 

类 型 包括 以 下 两 种 。 

。 强 引 用 

强 引用 是 默认 的 引用 类 型 。 被 强 引 用 指向 的 内 存 不 会 被 释放 。 强 引用 会 对 引用 计数 加 
1， 从 而 扩展 对 象 的 生命 周期 。 

。 3551 
弱 引 用 是 一 种 特殊 的 引用 类 型 。 它 不 会 增加 引用 计数 ， 因 而 不 会 扩展 对 象 的 生命 周期 。 
在 启用 了 ARC 的 Objective-C 编程 中 ， 弱 引用 格外 重要 。 
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其 他 类 型 的 引用 
Objective-C 当前 并 不 支持 其 他 类 型 的 引用 。 但 了 解 一 下 其 他 类 型 也 是 非常 有 趣 的 事情 。 
。 软 引 用 
软 引 用 与 弱 引 用 非常 相似 ， 只 是 前 者 没有 那么 迫切 地 抛弃 它 所 引用 的 对 象 。 如 果 一 
个 对 象 只 有 弱 引 用 存在 ， 那 么 这 个 对 象 会 在 下 个 垃圾 回收 周期 被 回收 ; 如果 一 个 对 
象 只 有 软 引 用 可 达 ， 那 么 这 个 对 象 一 般 还 能 再 坚持 一 会 。 
。 幽灵 引用 
这 是 力量 最 弱 的 引用 类 型 ， 会 被 最 早 地 回收 清理 。 幽 灵 引 用 的 对 象 与 已 回收 的 对 象 
比较 相似 ， 但 是 前 者 的 内 存 没 有 被 回收 利用 。 


这 些 引 用 类 型 没有 基于 引用 计数 系统 。 它 们 更 适合 用 于 垃圾 回收 系统 。 








2.6.1 变量 限定 符 
ARC 为 变量 供 了 四 种 生命 周期 限定 符 。 
e — strong 
这 是 默认 的 限定 符 ， 无 需 显 示 引 入 。 只 要 有 强 引用 指向 ， 对 象 就 会 长 时 间 驻 留 在 内 存 
中 。 可 以 将 strong 理解 为 retain 调用 的 ARC 版 本 。 
e _weak 
这 表明 引用 不 会 保持 被 引用 对 象 的 存活 。 当 没有 强 引 用 指向 对 和 象 时 ， 弱 引用 会 被 置 为 
nil, Al weak 看 作 是 assign 操作 符 的 ARC 版 本 ， 只 是 对 象 被 回收 时 ，_weak 具有 
安全 性 一 一 指针 将 自动 被 设置 为 nil。 
e __unsafe_unretained 
Lj _ weak 类 似 ， 只 是 当 没 有 强 引 用 指向 对 象 时 ，_unsafe_unretained 不 会 被 置 为 nil, 
可 将 其 看 作 assign 操作 符 的 ARC 版 本 。 
e — autoreleasing 
. autoreleasing 用 于 由 引用 使 用 id * 传递 的 消息 参数 。 它 预期 了 auterelease 方法 会 
在 传递 参数 的 方法 中 被 调用 。 
使 用 这 些 限 定 符 的 语义 如 下 : 
TypeName * qualifier variable; 


例 2-13 的 代码 展示 了 限定 符 的 使 用 。 
例 2-13 使 用 变量 限定 符 


Person * — strong p1 = [[Person alloc] init]; @ 

Person * — weak p2 = [[Person alloc] init]; @ 

Person * — unsafe unretained p3 = [[Person alloc] init]; © 
Person * — autoreleasing p4 = [[Person alloc] init]; @ 
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Q 创建 对 象 后 引用 计数 为 1， 并且 对 象 在 pl 引用 期 间 不 会 被 回收 。 
O 创建 对 象 后 引用 计数 为 0， 对 象 会 被 立即 释放 ， 且 p2 将 被 设置 为 nil。 





Q9 创建 对 象 后 引用 计数 为 1， 对象 会 被 立即 释放 ， 但 p3 不 会 被 设置 为 nil。 
O 创建 对 象 后 引用 计数 为 1， 当 方法 返回 时 对 象 会 被 立即 释放 。 


2.6.2 属性 限定 符 
属性 声明 有 两 个 新 的 持 有 关系 限定 符 : strong 和 weak。 此 外 ，assign 限定 符 的 语义 也 被 更 








ig 
É 
fr 
EE 








新 了 。 一 言 以 项 之 ,现在 共有 六 个 限定 符 。 





strong 

默认 符 ， 指定 了 _strong KA. 
weak 

指定 了 . weak 关系 。 

assign 


这 不 是 新 的 限定 符 ， 但 其 含义 发 生 了 改变 。 在 ARC Za, assign 是 默认 的 持 有 关系 限 
定 符 。 在 启用 ARC 之 后 ，assign 表示 了 | unsafe unretained KA, 


copy 

上 暗 指 了 _strong 关系。 此 外 ， 它 还 瞳 示 了 setter 中 的 复制 语义 (https://developer. 
apple.com/library/mac/documentation/Cocoa/reference/Foundation/Classes/nsobject_Class/ 
Reference/Reference.html#//apple_ref/occ/instm/NSObject/copy) 的 常规 行为 。 





retain 


指定 了 . strong 关系 。 


unsafe unretained 
指定 了 | unsafe unretained KA, 





例 2-14 展示 了 这 些 限定 符 。 因为 assign 和 unsafe unretained 只 进行 值 复制 而 没有 任何 
实质 性 的 检查 ， 所 以 它们 只 应 该 用 于 值 类 型 (B00L、NSInteger 、NSUInteger ， 等 等 ) MAE 
免 将 它们 用 于 引用 类 型 ， 尤 其 是 指针 类 型 ， 如 NSString * 和 UIView *, 


例 2-14 使 用 属性 限定 符 








@property (nonatomic, strong) IBOutlet UILabel *titleView; 
@property (nonatomic, weak) id<UIApplicationDelegate> appDelegate; 
@property (nonatomic, assign) UIView *danglingReference; @ 
@property (nonatomic, assign) BOOL selected; @ 

(property (nonatomic, copy) NSString *name; 

(property (nonatomic, retain) HPPhoto *photo; © 

@property (nonatomic, unsafe unretained) UIView *danglingReference; 


Q 错误 地 将 assign 用 于 指针 。 
O 对 值 类 型 正确 地 使 用 了 assign 限定 符 。 
© retain 是 ARC 纪元 之 前 的 老 古董 ， 现 代 的 代码 已 经 鲜 有 使 用 。 在 这 里 添加 它 只 是 为 了 











完整 性 。 
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2.7. KIDD 


现在 我 们 已 经 学 习 了 用 于 变量 和 属性 的 新 的 生命 周期 限定 符 。 让 我 们 素 
程 ， 然 后 观察 效果 。 


2.7.1 照片 模型 











自 实践 、 更 新 工 





创建 一 个 名 为 HPPhoto 的 类 ， 以 表示 相册 中 的 一 张 照 片 。 一 张 照片 包括 一 个 title、 一 个 


url 和 comments 列表 。 我 们 也 会 覆盖 dealloc 方法 以 观察 背后 的 秘密 。 
先 创建 一 个 新 的 Objective-C 类 : 

File 一 New — 10S — Cocoa Touch 一 Objective-C class 

例 2-15 给 出 了 典型 的 类 声明 。 


例 2-15  HPPhoto 类 


//HPPhoto.h 
@interface HPPhoto : NSObject 





(property (nonatomic, strong) HPAlbum *album; 
(property (nonatomic, strong) NSURL *url; 
(property (nonatomic, copy) NSString *title; 
(property (nonatomic, strong) NSArray *comments; 
Qend 


//HPPhoto.m 
@implementation HPPhoto 


-(void) dealloc 


DDLogVerbose(@"HPPhoto dealloc-ed"); 
} 


@end 


2.7.2 ”更 新 故事 板 


向 故事 板 中 的 第 一 个 视图 控制 器 添加 一 个 标签 和 四 个 按钮 。 这 些 按钮 将 触发 变量 的 创建 ， 








标签 用 于 显示 结果 。 最 终 呈现 的 UI 与 图 2-6 所 示 内 容 相 似 。 








> 图 Tab Bar Controller Scene 


v First View Controller - First Scene - 
Y © First View Controller - First 
(Ej Top Layout Guide 
B Bottom Layout Guide 
VView 7 H 
Cle - Fre View First View 
b |. Label - Result 
_ Button - Generate Crash 
..... Button ~ Create Strong Photo 
|_ Button - Create Weak Photo Result 
...; Button - Create Unsafe Unretained P... 
L_ Button - Create Strong > Weak Photo 


> (&) Constraints Generate Crash 
* Tab Bar Item - First 
® First Responder D Create Strong Photo 
Exit ZZ A 
= Va Create Weak Photo 
> Second View Controller - Second Scene 


Create Unsafe Unretained Photo 


Create Strong 一 Weak Photo 





S—— Ny 











图 2-6: 第 一 个 视图 控制 器 的 更 新 视图 


m 


| 


我 们 还 在 代码 中 添加 了 IBOutlet 和 IBAction 的 引用 ， 如 例 2-16 所 示 。 


例 2-16 HPFirstViewController.h 中 的 引用 更 新 
@interface HPFirstViewController : UIViewController 
@property (nonatomic, strong) IBOutlet UILabel *resultLabel; 


- (IBAction)createStrongPhoto:(id)sender; 

- (IBAction)createStrongToWeakPhoto: (id)sender; 

- (IBAction)createWeakPhoto: (id)sender; 

- (IBAction)createUnsafeUnretainedPhoto: (id)sender; 


@end 


2.7.8 方法 实现 
我 们 会 在 每 个 方法 中 做 如 下 事情 。 


(1) 创建 一 个 HPPhoto 类 的 实例 ， 并 将 此 实例 分 配给 一 个 本 地 引用 。 
(2) 设置 照片 的 title, 


(3) 在 resultLabel 中 显示 该 引用 是 否 为 niL。 如 果 不 是 nitL， 则 显示 title, 


























接 下 来 我 们 将 查看 每 个 方法 的 代码 〈 例 2-17 到 例 2-20)。 代 码 的 实现 大 致 相同 ， 唯 一 的 区 
别 在 于 创建 引用 的 类 型 。 注 意 ， 我 们 将 不 会 创建 方法 来 返回 引用 。 我 们 将 在 分 配 内 存 和 创 
建 引用 的 方法 内 探索 引用 。 我 们 还 尝试 用 NSLog 来 跟踪 生命 周期 事件 的 顺序 。 
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此 外 ， 我 们 还 研究 了 一 个 特殊 的 情况 ， 即 将 强 引 用 赋值 给 弱 引 用 ， 从 而 观察 对 象 会 发 生 什 


么 变化 。 


代码 的 结果 会 在 2.7.4 节 中 介绍 。 





























例 2-17 实现 createStrongPhoto 方法 
- (IBAction)createStrongPhoto: (id)sender 


{ 
DDLogDebug(@"%s enter", ^ PRETTY FUNCTION ); 
HPPhoto * — strong photo = [[HPPhoto alloc] init]; 
DDLogDebug(Q" Strong Photo: XQ", photo); 
photo.title = @"Strong Photo"; 


NSMutableString *ms = [[NSMutableString alloc] init]; 
[ms appendString: (photo == nil ? @"Photo is nil" : Q"Photo is not nil")]; 
[ms appendString:@"\n"]; 
if(photo != nil) { 
[ms appendString:photo.title]; 
} 
self.resultLabel.text = ms; 
DDLogDebug(@"%s exit", ^ PRETTY FUNCTION ); 
} 


例 2-18 ”实现 createWeakPhoto 方法 
- (IBAction)createWeakPhoto: (id)sender 


{ 
DDLogDebug(@"%s enter", ^ PRETTY FUNCTION ); 
HPPhoto * _ weak wphoto = [[HPPhoto alloc] init]; 
DDLogDebug(@"Weak Photo: %@", wphoto); 
wphoto.title = ("Weak Photo"; 


NSMutableString *ms - [[NSMutableString alloc] init]; 
[ms appendString:(wphoto == nil ? ("Photo is nil" : Q"Photo is not nil")]; 
[ns appendString:@"\n"]; 
if(wphoto != nil) { 
[ns appendString:wphoto.title]; 
} 
self.resultLabel.text = ms; 
DDLogDebug(@"%s exit", ^ PRETTY FUNCTION ); 
} 


例 2-19 实现 createStrongToWeakPhoto 方法 


-(void)createStrongToWeakPhoto: (id)sender 
{ 
DDLogDebug(@"%s enter", ^ PRETTY FUNCTION ); 
HPPhoto * sphoto - [[HPPhoto alloc] init]; 
DDLogDebug(Q" Strong Photo: %@", sphoto); 
sphoto.title = @"Strong Photo, Assigned to Weak"; 


HPPhoto * __weak wphoto = sphoto; 
DDLogDebug(@"Weak Photo: %@", wphoto); 


NSMutableString *ms = [[NSMutableString alloc] init]; 





[ms appendString:(wphoto == nil ? @"Photo is nil" : @"Photo is not nil")]; 
[ms appendString:@"\n"]; 


if(wphoto != nil) ( 


[ms appendString:wphoto. title]; 


j 


self.resultLabel.text = ms; 
DDLogDebug(@"%s exit", __PRETTY_FUNCTION_); 


例 2-20 ”实现 createUnsafeUnretainedPhoto 方法 


-(void)createUnsafeUnretainedPhoto:(id)sender 


( 


DDLogDebug(@"%s enter", ^ PRETTY FUNCTION ); 

HPPhoto * — unsafe unretained wphoto = [[HPPhoto alloc] init]; 
DDLogDebug(@"Unsafe Unretained Photo: %@", wphoto); 

wphoto.title - ("Strong Photo"; 


NSMutableString *ms - 


if(wphoto != nil) { 


[[NSMutableString alloc] init]; 
[ms appendString:(wphoto == nil ? @"Photo is nil" : @"Photo is not nil")]; 
[ms appendString:@"\n"]; 


[ms appendString:wphoto.title]; 


j 


self.resultLabel.text = ms; 
DDLogDebug(@"%s exit", __PRETTY_FUNCTION_); 


2.7.4 输出 分 析 





图 2-7 显示 了 输出 。 
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Flurry: Starting session on Agent Version [Flurry iOS 138 5.2.0] 
FV. Appear 


[D] [enter] createStrongPhoto 
[D] Strong Photo: «HPPhoto: 0x16dd0e30» 1 
[D] [exit] createStrongPhoto 

[V] HPVPhoto dealloc-ed 


[D] [enter] createWeakPhoto 
[V] HPVPhoto dealloc-ed 2 
[D] Weak Photo: (null) ) 

[D] [exit] createWeakPhoto 


[D] [enter] createStrongToWeakPhoto 
[D] Strong Photo: «HPPhoto: 0x16dc4020» 
[D] Weak Photo: «HPPhoto: 0x16dc4020» 


[D] [exit] createStrongToWeakPhoto 
[V] HPVPhoto dealloc-ed 


[D] [enter] createUnsafeUnretainedPhoto 
[V] HPVPhoto dealloc-ed 
[D] Unsafe Unretained Photo: «HPPhoto: 0x16ea26a0» 

[D] [exit] createUnsafeUnretainedPhoto 


[D] [enter] createUnsafeUnretainedPhoto 
[V] HPVPhoto dealloc-ed 


[D] Unsafe Unretained Photo: Unsafe Unretained Photo: 
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结果 不 言 自 明 ， 我 们 还 能 观察 到 一 些 有 趣 的 东西 。 


(1) strong 引用 (createStrongPhoto: 方法 ) 确保 了 对 象 在 其 作用 域内 不 会 被 销毁 。 对 
象 只 会 在 方法 完成 之 后 被 回收 。 

(2) _weak 引用 (createWeakPhoto: 方法 ) 对 引用 计数 没有 贡献 。 因 为 内 存 被 分 配 在 方法 
内 且 一 个 _weak 引用 指向 这 段 内 存 ， 所 以 引用 计数 为 0， 对 象 被 立即 回收 ， 甚 至 在 其 
被 用 于 紧邻 的 下 一 个 语句 前 。 

(3) 在 createStrongToWeakPhoto: 方法 中 ， 虽 然 weak 引用 不 会 增加 引用 计数 ， 但 之 前 创 

建 的 strong 引用 确保 了 对 象 不 会 在 方法 结束 前 释放 。 

(4) createUnsafeUnretainedPhoto: 方法 的 结果 更 加 有 趣 。 注 意 ， 对 象 会 立即 被 释放 ， 但 由 
于 内 存 还 没有 被 回收 ， 这 个 引用 可 以 使 用 ， 且 不 会 导致 错误 。 

(5) 但 是 ， 当 再 次 调用 该 方法 时 ， 我 们 不 仅 看 到 对 象 已 经 析 构 ， 而 且 内 存 也 被 重新 分 配 和 
再 使 用 了 。 于 是 ， 使 用 该 引用 导致 了 非法 访问 ， 应 用 出 现 了 以 SIGABRT 为 信号 的 崩溃 。 
这 是 由 内 存在 后 续 〈 对 象 析 构 之 后 ， 访 问 内 存 之 前 ) 被 回收 使 用 造成 的 。 

观察 图 2-8， 你 会 发 现 内 存 刚 好 在 设置 title 属性 前 被 回收 了 ， 从 而 导致 了 unrecognized 

selector sent to instance 错误 。 这 是 因为 内 存 已 经 被 回收 ， 并 且 现 在 可 能 已 经 用 于 存储 其 他 

对 象 。 






























































<UIKit/UIView.h> may also be helpful. 

2014-08-24 15:51:40.709 HPerf Apps[85481:60b] FV Appear 

2014-08-24 15:51:43.512 HPerf Apps[85481:60b] FV Ph UU 

2014-08-24 15:51:43.513 HPerf Apps[85481:60b] [D] [enter] createUnsafeUnretainedPhoto 
2014-08-24 15:51:43.513 HPerf Apps[85481:60b] [V] HPVPhoto dealloc-ed 

2014-08-24 15:51:43.513 HPerf Apps[85481:60b] [D] Unsafe Unretained Photo: «HPPhoto: 
0x10ea32180» 

2014-08-24 15:51:43.513 HPerf Apps[85481:60b] -[__NSCFString setTitle:]: unrecognized 
selector sent to instance 0x10ea32180 














2-8; — unsafe unretained 导致 的 崩 演 


2.8 僵尸 对 象 

僵尸 对 象 是 用 于 捕捉 内 存 错 误 的 调试 功能 。 

通常 情况 下 ， 当 引用 计数 降 为 0 时 对 象 会 立即 被 释放 ， 但 这 使 得 调试 变 得 困难 。 如 果 开 局 
了 僵尸 对 象 ， 那 么 对 象 就 不 会 立即 释放 内 存 ， 而 是 被 标记 为 僵尸 。 任 何 试图 对 其 进行 访 
问 的 行为 都 会 被 日 志 记 录 ， 因 而 你 可 以 在 对 象 的 生命 周期 中 跟踪 对 象 在 代码 中 被 使 用 的 
位 置 。 

NSZombieEnabled 是 一 个 环境 变量 ， 可 以 控制 Core Foundation 的 运行 时 是 否 将 使 用 僵尸 对 
象 。 不 应 长 期 保留 NSZzombieEnabLed， 因 为 默认 情况 下 不 会 有 对 象 被 真正 析 构 ， 这 会 导致 应 
用 使 用 大 量 的 内 存 。 特 别 说 明 一 点 ， 在 发 布 的 构建 包 中 一 定 要 禁用 NSzombteEnabled, 

要 想 设置 NSZombieEnabled 环境 变量 ， 需 要 进入 Product > Scheme 一 Edit Scheme。 选 择 
左 侧 的 Ruan， 然后 在 右 侧 选 取 Diagnostics 标签 页 。 选 中 Enable Zombie Objects 选项 ， 如 图 
2-9 所 示 。 
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Duplicate Scheme Manage Schemes... Shared Close 
图 2-9: 在 XCode 设置 中 开启 僵尸 对 象 
Par’ 
2.9 内 存 管理 规则 
现在 我 们 已 经 了 解 了 生命 周期 限定 符 的 细节 ， 复 习 一 下 内 存 管理 的 基本 规则 非常 重要 。 























正如 苹果 公司 的 官方 文档 所 述 ， 内 存 管 理 有 四 个 基本 规则 。 








你 拥有 所 有 自己 创建 的 对 象 ， 如 new, alloc, copy 或 mutableCopy, 
你 可 以 用 MRC 中 的 retain 或 者 ARC 中 的 strong 引用 来 拥有 任何 对 象 的 持 有 关系 。 
在 MRC 中 ， 当 不 再 需要 某 个 对 象 时 ， 你 必须 立即 使 用 release 方法 来 放弃 对 该 对 象 的 


持 有 关系 。 而 在 ARC 中 则 无 需 任 何 特殊 操作 。 持 有 关系 会 在 对 象 失去 最 后 的 引用 (An 





方法 中 的 最 后 一 行 代码 ) 时 被 抛弃 。 





一 定 不 能 抛弃 原本 并 不 存在 持 有 关系 的 对 象 。 


要 想 避 免 内 存 泄 漏 和 应 用 崩 漠 ， 你 应 当 在 编写 Objective-C 代码 时 牢记 这 些 规则 。 


2.10 循环 引用 


引用 计数 的 最 大 陷阱 在 于 ， 它 不 能 处 型 
将 在 本 节 中 讨论 循环 引用 出 现 的 典型 场景 ， 堵 





E 环 状 的 引用 关系 ， 即 Objective-C 的 循环 引用 。 我 们 


F 介 绍 避 免 循 环 引 用 的 最 佳 实践 。 
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如 果 仔 细 学 习 了 上 一 节 描 述 的 规则 ， 你 会 发 现 它 们 不 过 是 引用 计数 的 实现 。 声 明 持 有 关系 
会 增加 引用 计数 ， 而 放弃 持 有 关系 则 会 减少 引用 计数 。 当 引用 计数 降 为 0 时， 系统 将 回收 














在 我 们 的 示例 应 用 中 ，HPALbun 实体 包含 coverPhoto 和 photos 数组 ， 以 表示 相册 的 封面 照 
片 以 及 与 之 关联 的 照片 。 类 似 地 ， 除 了 其 他 的 属性 (如 URL、 标 题 、 评 论 等 )，HPPhoto 可 
能 还 代表 一 张 照片 属于 某 个 相册 。 例 2-21 展示 了 定义 这 一 实体 信息 的 代表 性 代码 。 





例 2-21 循环 引用 
@class HPPhoto; 
@interface HPAlbum : 
(property (nonatomic, 
(property (nonatomic, 
(property (nonatomic, 


@property (nonatomic, 


@end 


@interface HPPhoto : 


@property (nonatomic, 
@property (nonatomic, 
@property (nonatomic, 
@property (nonatomic, 


@end 














NSObject 


copy) NSString *name; 

strong) NSDate *creationTime; 
copy) HPPhoto *coverPhoto; @ 
copy) NSArray *photos; @ 


NSObject 


strong) HPAlbum *album; © 
strong) NSURL *url; 

copy) NSString *title; 
copy) NSArray *comments; 


@ HPAlbum 对 coverPhoto 有 一 个 强 引用 ， 类 型 为 HPPhoto。 
@ 它 通 过 photos 数组 还 持 有 了 许多 其 他 的 HPPhoto 对 象 。 
© HPPhoto 通过 强 引用 指向 了 它 所 属 的 相册 。 


为 了 简化 讨论 ， 我 们 假设 一 个 相册 中 包含 两 张 照片 : pl (相册 封面 ) 和 p2。 引 用 计数 如 下 。 


。 pl 在 photos 和 coverPhoto 中 有 强 引 用 。 引 用 计数 为 2。 
e p2 在 photos 中 有 强 引 用 。 引 用 计数 为 1。 
e album Æ p1 和 p2 中 有 强 引用 。 引 用 计数 为 2。 


我 们 在 前 面 的 2.6 市 中 讨论 过 强 引 用 。 
这 些 对 象 通过 名 为 createAlbun 的 方法 被 创建 。 虽 然 这 些 对 象 从 某 个 时 间 点 后 不 再 被 使 用 ， 





但 它们 的 内 存 不 会 被 释放 ， 



























































因为 它们 的 引用 计数 都 不 会 降 为 0。 图 2-10 演示 了 这 种 关系 。 



















[was e. s mis o mas 
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pl Album p2 











图 2-10: 相册 与 照片 的 引用 


2.10.1 避免 循环 引用 的 规则 


上 一 太 演 示 了 发 生 循环 引用 的 场景 。 在 本 市 中 ， 我 们 将 关注 避免 在 代码 中 发 生 循环 引用 的 





规则 。 








。 对 象 不 应 该 持 有 它 的 父 对 象 ， 应 该 用 weak 引用 指向 它 的 父 对 象 ( 见 2.6 市 )。 
在 上 一 个 场景 中 ， 照 片 被 atbum 所 包含 ， 我 们 可 以 将 照片 看 成 孩子 。 因 此 ， 从 照片 到 相 





册 的 引用 应 该 是 弱 引 用 。 弱 引用 对 引用 计数 没有 贡献 。 
更 新 后 的 引用 计数 如 下 。 


(1) p1 通过 photos 和 coverPhoto 被 强 引 用 。 引 用 计数 为 2。 
(2) p2 通过 photos 被 强 引 用 。 引 用 计数 为 1。 
(3) album 不 存在 任何 强 引用 。 引 用 计数 为 0。 














会 下 降 为 0， 于 是 它们 也 会 被 回收 。 
。 作为 必然 的 结果 ， 一 个 层级 体系 中 的 子 对 象 应 该 保留 祖先 对 象 。 








因此 ， 当 不 再 被 使 用 ，albunm 对 象 会 被 回收 。 一 旦 album 被 释放 ，p1 和 p2 的 引用 计数 





。 连接 对 象 不 应 持 有 它们 的 目标 对 象 。 目标 对 象 的 角色 是 持 有 者 。 连接 对 象 包括 以 下 几 种 。 


(1) 使 用 委托 的 对 象 。 委 托 应 该 被 当 作 目标 对 象 ， 即 持 有 者 。 




















(2) 包含 目标 和 action 的 对 象 ， 这 是 由 上 一 条 规则 推理 得 到 的 。 例 如 ，UIButton 会 调用 








它 的 目标 对 象 上 的 action 方法 。 按 钮 不 应 该 保留 它 的 目标 。 


(3) 观察 者 模式 中 被 观察 的 对 象 。 观 察 者 就 是 持 有 者 ， 并 会 观察 发 生 在 被 观察 对 象 上 的 


变化 。 
。 使 用 专用 的 销毁 方法 中 断 循 环 引 用 。 
双向 链表 中 存在 循环 引用 ， 环 形 链表 中 也 存在 循环 引用 。 








在 这 类 情况 下 ， 一 旦 明确 对 象 不 会 再 被 使 用 时 〈 当 链表 的 表 头 超出 作用 范 

















围 )， 你 要 编 





写 代码 以 打破 链表 的 链接 。 创 建 一 个 (名 为 delink 的 ) 方法 切断 其 自身 与 链表 中 下 一 





个 而 点 的 链接 。 通 过 访问 者 模式 递归 地 执行 这 一 过 程 ， 从 而 避免 无 限 递 归 。 
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2.10.2 循环 引用 的 常见 场景 

大 把 的 常见 场景 会 导致 循环 引用 。 例 如 ， 使 用 线程 、 计 时 器 、 简 单 的 块 方法 或 委托 都 可 能 
会 导致 循环 引用 。 接 下 来 我 们 将 逐步 探索 这 些 场景 ， 并 给 出 避免 循环 引用 的 步骤 。 

1. 委托 

委托 很 可 能 是 引入 循环 引用 的 最 常见 的 地 方 。 在 应 用 局 动 时 ， 从 服务 器 获取 最 新 的 数据 并 
更 新 UI 是 常见 的 事情 。 当 用 户 点 击 刷新 按钮 时 也 会 触发 类 似 的 刷新 逻辑 。 
考虑 这 个 特定 的 场景 : 展示 了 一 组 记录 的 视图 控制 器 ， 以 及 一 个 用 户 点 击 后 就 会 刷新 列表 
的 刷新 按钮 。 

我 们 在 实现 方面 用 了 两 个 类 : HPDataListViewController 用 于 UI, HPDataUpdateOp 用 于 模 
拟 对 网 络 的 调用 。 例 2-22 展示 了 视图 控制 器 的 代码 ， 例 2-23 展示 了 更 新 操作 的 代码 。 


例 2-22 应 用 更 新 调用 






































//HPDataListViewController.h 
@interface HPDataListViewController : UIViewController Qj 


@property (nonatomic, strong) HPDataUpdateOp *updateOp; €) 
(property (nonatomic, strong) BOOL refreshing; 


- (IBAction)onRefreshClick: (id)sender; 
@end 


//HPDataListViewController.m 
(implementation HPDataListViewController 


// 为 达到 简洁 的 目的 ,删除 viewDidLoad 的 代码 





- (IBAction)onRefreshClicked:(id)sender { © 
DDLogDebug(@"%s enter", ^ PRETTY FUNCTION 5; @ 
if([self.refreshing == NO]) { 

self.refreshing - YES; 
if(self.updateOp == nil) { 
[self.updateOp - [[HPDataUpdateOp new]; 
} 
[self.updateOp startWithDelegate:self 
withSelector :@selector(onDataAvailable:)]; © 
} 
DDLogDebug(@"%s exit", ^ PRETTY FUNCTION ); (4) 
} 


- (void)onDataAvailable:(NSArray *)records { © 
// 用 最 新 记录 更 新 UI 
self.refreshing = NO; 
self.updateOp - nil; 

} 





@end 





@ HPDataListviewController 在 列表 中 显示 数据 。 

Q updatedp 实现 拉 取 数据 的 网 络 操作 。 

Q 用 户 点 击 refreshButton 时 会 调用 这 个 方法 。 

O 记录 日 志 以 监控 执行 顺序 。 

e peni 方法 可 以 在 结果 可 用 时 调用 回调 方法 。 

Q onDataAvailable 是 回调 方法 。 它 会 更 新 视图 控制 器 的 状态 和 UI。 


例 2-23 更 新 操作 


//HPDataUpdateOp.m 
@implementation HPDataUpdateOp 








-(void)startWithDelegate:(id)delegate withSelector:(SEL)selector { 
dispatch async( 
dispatch get global queue(DISPATCH QUEUE PRIORITY DEFAULT, 0), ^( @ 
// 执 行 某 个 操作 @ 
dispatch async(dispatch get main queue(), ^( © 
if([delegate respondsToSelector:selector]) { 
[delegate performSelector:selector 
withObject:[NSArray arrayWithObjects:nil]; @ 


335 
D; 
} 


-(void)dealloc { 
DDLogDebug(@"%s called", __PRETTY_FUNCTION_); 


} 
@end 


Q 所 有 需要 长 时 间 运 行 的 任务 都 应 该 在 主线 程 之 外 进行 。 
e 假设 操作 需要 消耗 2 秒 。 

© 一 旦 结果 可 用 ， 将 上 下 文 切换 回 主线 程 
@.… 调用 选择 器 


onRefreshClicked 方法 将 self f£ A updateOp 方法 。 同 时，HPDataListViewController 持 有 
了 对 updateop 的 引用 。 这 就 是 产生 循环 引用 的 地 方 。 


就 解决 方案 而 言 ， 其 中 一 个 选择 是 不 要 将 updateop 作为 一 个 属性 ， 而 在 onRefreshcticked: 
方法 中 创建 HPDataUpdateOp 的 一 个 实例 。 此 时 ，update0p 会 持 有 HPDataListViewController 
对 象 的 引用 ， 但 反 过 来 不 会 。 更 新 后 的 代码 如 例 2-24 所 示 。 


例 2-24 在 无 需 属性 的 情况 下 更 新 应 用 


- (IBAction)onRefreshClicked:(id)sender { 
DDLogDebug(@"%s enter", ^ PRETTY FUNCTION ); 
if(self.refreshing == NO) { 
self.refreshing = YES; 
HPDataUpdateOp *updateOp = [[HPDataUpdateOp new]; @ 
[updateOp startWithDelegate:self withSelector:@selector(onDataAvailable: )]; 





= 
Hn 
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DDLogDebug(@"%s 
} 


exit", —PRETTY FUNCTION ); 


Q 创建 本 地 变量 而 不 再 持 有 引用 。 


这 确实 解决 了 引入 循环 ? 











| 用 的 问题 ， 但 却 带 来 了 其 他 问题 。updateOp 对 象 永远 不 会 被 其 他 








地 方 所 引用 ， 因 此 ， 一 旦 onRefreshClicked: 方法 执行 完毕 ， 它 的 引用 计数 将 降 为 0， 然后 
立即 被 回收 。 输 出 如 图 2-11 所 示 。 




















2014-09-01 22:59:43.330 HPerf Apps[29264:60b] [D] [onRefreshClicked] enter 
2014-09-01 22:59:43.331 HPerf Apps[29264:60b] [D] [onRefreshClicked] exit 
2014-09-01 22:59:43.332 HPerf Apps[29264:60b] [D] [FriendsUpdateOp::dealloc] called 








2-11: 使 用 本 地 变量 的 输出 结果 


正如 演示 所 示 ，HPDataUp 


dateOp 是 高 度 简化 的 场景 。 通 常 来 说 ， 应 用 会 通过 网 络 队 列 来 实 








现 排队 执行 更 新 操作 。 二 


F 且 在 操作 完成 之 后 ， 用 户 很 有 可 能 会 切换 到 其 他 的 视图 控制 器 。 























在 这 种 场景 中 ， 视 图 控制 器 在 理想 的 情况 下 应 立即 被 回收 。 但 因为 视图 控制 器 正 被 网 络 操 
作 所 使 用 ， 所 以 它 不 会 被 回收 。 现 在 想象 一 下 多 个 视图 控制 器 被 队列 中 的 操作 所 持 有 。 虽 
然 没 有 造成 循环 引用 ， 但 确实 会 增加 内 存 峰 值 需求 。 因 此 ， 这 富 无 疑问 是 个 bug， 因 为 如 





















































果 没 有 视图 控制 器 的 存在 ， 操 作 在 理想 状态 下 应 该 及 时 地 释放 对 象 。 
因此 ， 严 格 来 讲 ， 这 个 修复 并 没有 解决 问题 。 原 因 是 什么 呢 ? 问题 在 于 HPDataUpdateOp f 


有 了 HPDataListViewCont 


这 些 对 象 之 间 完 全 失去 链接 。 








roller 对 象 的 强 引 用 。 但 这 里 也 不 能 使 用 弱 引 用 ， 因 为 这 会 导致 











解决 方案 是 在 委托 (当前 示例 为 视图 控制 器 ) 中 建立 对 操作 的 强 引 用 ， 并 在 操作 中 建立 对 


委托 的 弱 引 用 。 
当 准 备 调用 回调 方法 时 ， 




















操作 应 当 获 取 委托 的 的 强 引 用 。 





不 仅 如 此 ， 我 们 还 应 该 在 HPDataUpdateOp 中 引入 cancel 方法 ， 以 便 在 视图 控制 器 即将 被 
回收 时 可 以 调用 cancel 方法 。 例 2-25 展示 了 实现 这 一 效果 的 更 新 代码 。 


例 2-25 HPDataListViewController 和 HPDataUpdateOp 的 最 终 版 本 


//HPDataListViewController 
- (IBAction)onRefreshClicked:(id)sender { 
DDLogDebug(@"%s enter", X PRETTY FUNCTION ); 
self.updateOp = [[HPDataUpdateOp new]; @ 
[self.updateOp startUsingDelegate:self 
withSelector :@selector(onDataAvailable: )]; 
DDLogDebug(@"%s exit", __PRETTY_FUNCTION_); 


J 


-(void)onDataAvailable:(NSArray *)records { 
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DDLogDebug(@"%s called", _ PRETTY FUNCTION ); 
self.resultLabel.text = @"[- onDataAvailable] called"; 
self.updateOp = nil; € 

} 


-(void)dealloc { 
DDLogDebug(@"%s called", _ PRETTY FUNCTION ); 
if(self.updateOp != nil) { 
[self.updateOp cancel]; © 
} 
} 


/ [HPDataUpdateOp.h 
(protocol HPDataUpdateOpDelegate <NSObject> 


-(void)onDataAvailable:(NSArray *)records; 

Qend 

@interface HPDataUpdateOp 

@property (nonatomic, weak) id<HPDataUpdateOpDelegate> delegate; @ 


-(void)startUpdate; 
-(void)cancel; 


Qend 


/|HPDataUpdate0p.m 
(implementation HPDataUpdateOp 
-(void)startUpdate { 
dispatch async( 
dispatch get global queue(DISPATCH QUEUE PRIORITY DEFAULT, 0), “{ 
// 执 行 网 络 调用 ,然后 报告 结果 
//NSArray *records =... 
dispatch_async(dispatch_get_main_queue(), “{ 
id«HPDataUpdateOpDelegate» delegate = self.delegate; @ 
if(!delegate) { © 





return; 


} else { @ 


[delegate onDataAvailable: records]; 
} 


p); 
p); 
} 


-(void)cancel { Q 

// 取 消 执行 中 的 网 络 请 求 
self.delegate = nil; 

} 


@ 使 用 属性 表示 操作 。 视 图 控制 器 圭 有 操作 。 
O 当 任务 完成 后 将 属性 设置 为 nil。 实现 对 操作 对 象 的 回收 。 
© 若 视 图 控制 器 即将 被 回收 ， 则 取消 操作 。 











L 
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O 操作 保持 对 回调 委托 的 弱 引 用 
尝试 获取 委托 的 强 引 用 。 

Q 如 果 原 始 对 象 仍然 存在 …… 

(7 - 通过 它 报告 onDataAvailable。 

Q 取消 操作 显 式 地 要 求 废 弃 回 调 对 象 。 

本 质 上 ， 这 里 实现 了 2.10.1 节 中 的 第 一 条 规则 。HPDataListViewController 是 持 有 者 ， 而 

HPDataUpdateOp 是 被 持 有 的 对 象 (如 持 有 层级 中 的 子 节 点 )。 


2-12 展示 了 网 络 响应 早 于 用 户 离开 页 面 的 结果 。 图 2-13 展示 了 网 络 响应 晚 于 用 户 离开 
页 面 时 的 输出 信息 。 


o 




















2014-09-01 23:02:25.396 HPerf Apps[29283:60b] [D] [onRefreshClicked] enter 
2014-09-01 23:02:25.398 HPerf Apps[29283:60b] [D] [onRefreshClicked] exit 
2014-09-01 23:02:27.400 HPerf Apps[29283:60b] [D] [onFriendsAvailable] called 
2014-09-01 23:02:27.401 HPerf Apps[29283:60b] [D] [FriendsUpdateOp::dealloc] called 








图 2-12: 操作 完成 后 ， 视 图 控制 器 可 用 时 使 用 更 新 代码 的 结果 





2014-09-01 23:47:35.490 HPerf Apps[29283:60b] [D] [onRefreshClicked] enter 

2014-09-01 23:47:35.491 HPerf Apps[29283:60b] [D] [onRefreshClicked] exit 

2014-09-01 23:47:36.586 HPerf Apps[29283:60b] Appear Chxx 

2014-09-01 23:47:36.587 HPerf Apps[29283:60b] [D] [HPFriendsListViewController::dealloc] called 
2014-09-01 23:47:37.493 HPerf Apps[29283:60b] [D] [FriendsUpdateOp::start] [dispatch async::main] 
delegate is nil 

2014-09-01 23:47:37.494 HPerf Apps[29283:60b] [D] [FriendsUpdateOp::dealloc] called 











图 2-13: 操作 完成 前 ， 视 图 控制 器 已 销毁 时 使 用 更 新 代码 的 结果 





虽然 看 起 来 似乎 比较 直观 ， 但 随 着 执行 深入 到 不 同 的 层 而 产生 复杂 的 对 象 图 时 ， 情 况 会 变 
得 非常 复杂 。 你 需要 确保 自己 不 会 在 网 络 的 底层 、 数 据 库 以 及 用 于 UI EE (如 创建 对 象 
的 层 ) 的 存储 中 持 有 引用 。 

2. 块 

与 不 正确 地 使 用 委托 对 象 导致 的 问题 类 似 ， 在 使 用 块 时 ， 捕 获 外 部 变量 也 是 导致 循环 引用 


参考 例 2-26 中 的 简单 代码 。 
例 2-26 使 用 块 捕获 变量 


-(void)someMethod { 
SomeViewController *vc = [[SomeViewController alloc] init]; 
[self presentViewController:vc animated:YES 
completion:^[ 





























50 | #22 


self.data - vc.data; 
[self dismissViewControllerAnimated:YES completion:nil]; 
)]; 
} 


iE RY fie. SCRE HE ER A BY et S — — 3 0E f fl ea S SO Be, A) A EE a 
用 户 显 示 它 ， 其 父 视图 控制 器 也 不 会 被 回收 ， 因 为 它 被 comptetion 块 捕获 了 。 在 
SoneViewController 执行 耗 时 较 长 的 任务 时 ， 如 图 像 处 理 或 复杂 的 视图 泻 染 ， 其 父 视图 控 
制 器 的 内 存 不 会 被 清空 ， 应 用 存在 内 存 不 足 的 风险 。 

例 2-27 所 示 的 解决 方案 与 我 们 在 前 一 节 讨 论 的 内 容 相似 。 


例 2-27 使 用 块 时 的 变量 捕获 
-(void)someMethod { 
SomeViewController *vc = [[SomeViewController alloc] init]; 




























































































. weak typeof(self) weakSelf = self; @ 


[self presentViewController:vc animated: YES 
completion:^[ 
typeof(self) theSelf = weakSelf; @ 


if(theself != nil) ( © 
theSelf.data = vc.data; @ 
[theSelf dismissViewControllerAnimated:YES completion:nil]; 
} 
FI; 
} 
Q 获得 一 个 弱 引 用 。 
O 通过 弱 引 用 获得 强 引用 。 注 意 ，_strong 是 隐 式 的 ， 可 以 增加 引用 计数 …… 
© …… 只 在 不 为 ntt 时 才 继 续 …… 
o: 处 理 后 续 操作 。 
3. 线程 与 计时 器 
不 正确 地 使 用 NSThread 和 NSTimer 对 象 也 可 能 会 导致 循环 引用 。 运 行 异 步 操作 的 典型 步骤 
如 下 。 


。 如 果 没 有 编写 更 高 级 的 代码 来 管理 自 定义 的 队列 ， 则 在 全 局 队列 上 使 用 dispatch async 
方法 。 

。 在 需要 的 时 间 和 地 点 用 NSThread 开启 异步 执行 。 

。 使 用 NSTimer 周期 性 地 执行 一 段 代码 。 


假设 有 一 个 新 闻 应 用 ， 其 UI 显示 已 登录 用 户 的 新 闻 流 ， 且 该 应 用 每 隔 2 分 钟 自 动 刷新 一 次 。 
例 2-28 展示 了 执行 周期 性 更 新 的 典型 代码 。 
例 2-28 使 用 NSTimer 


(implementation HPNewsFeedViewController 
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-(void)startCountdown { 

self.timer = [NSTimer scheduledTimerWithTimeInterval:120 
target:self 
selector :@seLector(updateFeed: ) 
userInfo:nil repeats:YES]; 


} 


-(void)dealloc { 
[self.timer invalidate]; 


} 

@end 
例 2-28 中 的 循环 引用 非常 明显 一 一 对 象 持 有 了 计时 器 ， 同 时 计时 器 也 持 有 了 对 象 。 与 例 
2-22 类 似 ， 我 们 无 法 通过 “消灭 ”属性 来 解决 问题 。 事 实 上 ， 我 们 需要 持 有 timer 属性 ， 
以 便 其 可 以 在 后 续 被 销毁 。 
对 这 段 代 码 来 说 ， 运 行 循环 也 将 持 有 计时 器 ， 并 直到 invalidate 方法 被 调用 时 才 会 释放 它 。 
这 就 创建 了 对 计时 器 对 象 的 附加 持 有 引用 ， 即 使 代码 中 并 疫 有 显 式 的 引用 关系 ， 这 仍然 会 
导致 循环 引用 。 
NSTimer 对 象 导致 了 被 运行 时 持 有 的 间接 引用 。 这 些 引用 是 强 引用 ， 因 而 目 
标的 引用 计数 会 以 2 (而 不 是 1) 增长 。 必 须 对 计时 器 对 象 调用 invalidate, 
以 移 除 引 用 。 





























假设 例 2-28 中 的 代码 属于 一 个 视图 榨 制 器 ， 并 且 由 于 用 户 的 操作 ， 视 图 控制 器 在 应 用 中 被 
创建 了 多 次 。 可 以 想象 内 存 泄 漏 的 总 量 会 有 多 么 大 。 

如 果 你 使 用 的 是 NSThread， 那 也 不 要 得 意 ， 因 为 同样 的 问题 还 是 会 发 生 。 这 个 问题 有 两 个 
解决 方法 。 

。 主动 调用 invalidate。 

。 将 代码 分 离 到 多 个 类 中 。 

接 下 来 我 们 将 分 别 讨论 这 两 种 解决 方法 。 

别 指 望 dealloc 能 够 清理 这 些 对 象 。 为 什么 呢 ? 如 果 建 立 了 循环 引用 ， 那 dealloc 方法 永 
远 都 不 会 被 调用 ， 计 时 器 也 永远 都 不 会 执行 invalidated。 因 为 运行 循环 会 跟踪 活跃 的 计 
时 器 对 象 和 线程 对 象 ， 所 以 仅 在 代码 中 置 为 nil 并 不 能 销毁 对 象 。 要 想 解 决 这 个 问题 ， 可 
以 创建 一 个 自 定义 方法 ， 以 更 加 明确 的 方式 执行 清理 操作 。 

在 一 个 视图 控制 器 中 ， 调 用 这 个 清理 方法 的 最 佳 时 机 是 用 户 离开 视图 控制 器 的 时 候 ， 这 个 
时 机 既 可 以 是 点 击 返 回 按钮 ， 也 可 以 是 其 他 类 似 的 行为 (类 知道 此 事 发 生 的 地 方 )。 我 们 
将 这 个 方法 命名 为 cleanup。 例 2-29 提供 了 实现 的 代码 。 











































































































例 2-29 清理 NSTimer 
-(void)didMoveToParentViewController:(UIViewController *) parent ( @ 
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if(parent == nil) { 
[self cleanup]; 

} 

} 


-(void)cleanup { 
[self.timer invalidate]; 


} 





Q 当 视图 控制 器 进入 或 离开 父 视图 控制 器 上 时， 调用 didMoveToParentViewController 方法 。 
图 控制 器 离 


在 

















例 2-29 中 ， 通 过 重 写 didMoveToParentViewController 方法 ， 当 用 户 从 父 视 
开 当 前 视图 控制 器 时 ， 我 们 执行 清理 操作 。 这 要 比 调用 dealloc 更 加 明确 。 





























另 一 种 方法 是 修改 返回 按钮 的 目标 ， 如 例 2-30 所 示 。 


例 2-30 通过 拦截 返回 按钮 执行 清理 


Q 拦截 导航 控制 器 对 返回 按钮 的 点 击 事 
O 在 视图 控制 器 弹出 之 前 进行 清理 。 





-(id)init { 
if(self = [super init]) { 
self.navigationItem.backBarButtonItem.target - self; 
self.navigationItem.backBarButtonItem.action 
= @selector(backButtonPressDetected:); @ 
} 
return self; 


} 


-(void)backButtonPressDetected:(id)sender { 
[self cleanup]; @ 
[self .navigationController popViewControllerAnimated: TRUE]; 


} 





Dr 
Tr 
o 



































另 一 个 清理 方案 是 将 持 有 关系 分 散 到 多 个 类 中 一 一 任务 类 执行 具体 动作 ， 所 有 者 类 调用 
任务 。 


我 更 推荐 后 一 个 方案 ， 理 由 如 下 。 


我 们 可 























清理 器 有 定义 良好 的 职责 持 有 者 。 
需要 时 任务 可 以 被 多 个 持 有 者 重复 使 用 。 














HPNewsFeedUpdateTask 周期 性 地 执行 ， 检 查 填 充 视 图 控制 器 的 最 新 的 feed 流 。 
要 实现 这 个 效果 ， 重 构 后 的 代码 如 例 2-31 所 示 。 


例 2-31 使 用 计时 器 重 构 后 的 代码 


//HPNewsFeedUpdateTask.h 
@interface HPNewsFeedUpdateTask 


@property (nonatomic, weak) id target; @ 
@property (nonatomic, assign) SEL selector; 


以 将 前 面 的 代码 拆 成 两 个 类 : Be e 展示 最 新 的 feed 流 ， 而 
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Qend 


/ | HPNewsFeedUpdateTask.m 
(implementation HPNewsFeedUpdateTask 


-(void)initWithTimeInterval:(NSTimeInterval)interval 
target:(id)target selector:(SEL)selector { @ 


if(self = [super init]) { 
self.target - target; 
self.selector - selector; 


self.timer - [NSTimer scheduledTimerWithTimeInterval:interval 
target:self selector :@selector(fetchAndUpdate: ) 
userInfo:nil repeats: YES]; 


j 


return self; 


j 


-(void)fetchAndUpdate:(NSTimer *)timer ( © 


// 检 索 feed 
HPNewsFeed *feed = [self getFromServerAndCreateModel]; 


. weak typeof(self) weakSelf = self; @ 


dispatch async(dispatch get main queue(), “{ 
. strong typeof(self) sself - weakSelf; 
if(!sself) { 
return; 


j 


if(sself.target == nil) { 
return; 


j 


id target = sself.target; @ 
SEL selector - sself.selector; 


if([target respondsToSelector:selector]) { 
[target performSelector:selector withObject:feed]; 


p; 
} 


-(void)shutdown ( @ 
[self.timer invalidate]; 
self.timer - nil; 

} 

@end 


//HPNewsFeedViewController.m 
(implement HPNewsFeedViewController 


-(void)viewDidLoad { @ 
self.updateTask = [HPNewsFeedUpdateTask initWithTimeInterval:120 





邮 


target:self selector :@selector(updateUsingFeed:)]; 


} 


-(void)updateUsingFeed:(HPNewsFeed *)feed { @ 
// 更 新 UI 
} 


-(void)dealloc { © 
[self.updateTask shutdown]; 

} 

@end 


接 下 来 我 们 看 一 下 对 HPNewsFeedUpdateTask 的 详细 分 析 。 


(1) target 属性 @ 25555 LH. target 会 在 这 里 实例 化 任务 并 桂 有 它 。 

(2) initWithTimeInterval:@ 是 推荐 使 用 的 方法 。 它 需要 一 些 必 要 的 输入 ， 并 启动 计时 器 。 
(3) fetchAndUpdate: 方法 © 会 周期 性 地 执行 。 

(4) 在 使 用 异步 块 时 需要 确保 不 会 引入 循环 引用 。 我 们 在 块 方法 内 使 用 _weak 引用 0. 

(5) 在 fetchAndUpdate: 方法 @ 中 ，target 和 selector 的 本 地 变量 都 在 调用 respondsToSelector: 

前 创建 ， 并 执行 操作 。 

这 样 做 是 为 了 避免 在 以 下 的 执行 序列 中 发 生 竞争 情况 。 

a. 在 某 个 线程 A 中 调用 [target respondsToSelector:selector], 

b. 在 线程 B 中 修改 target 或 selector, 

c. 在 线程 A 中 调用 [target performSelector:selector withobject:feed]。 有 了 这 个 代 
码 ， 即 使 target sk selector 此 刻 已 经 发 生 改 变 ，performSelector 仍然 会 被 正确 的 
target 和 selector 所 调用 。 

(6) shutdown 方法 Q 对 计时 器 调用 invalidate。 运 行 循环 会 终止 对 计时 器 的 调用 ， 于 是 计 

时 器 成 为 任务 对 象 持 有 的 唯一 引用 。 

从 使 用 方面 来 看 ，HPNewsFeedViewController 使 用 了 HPNewsFeedUpdateTask。 控 制 器 没有 

被 除 父 控制 器 之 外 的 对 象 所 持 有 。 因 此 ， 当 用 户 离开 页 面 时 (也 就 是 点 击 了 返回 按钮 时 )， 

引用 计数 会 降 为 0， 视 图 控制 器 会 被 销毁 。 这 反 过 来 会 导致 更 新 任务 停止 ， 进 而 导致 计时 

器 被 设 定 无 效 ， 从 而 触发 所 有 关联 对 象 (包括 timer 和 updateTask) 的 析 构 链 。 

现在 我 们 对 例 2-31 中 的 HPNewsFeedViewController 代码 进行 分 析 。 

(1) 在 viewDidLoad 方法 @ 中 ， 对 任务 对 象 进行 初始 化 ， 其 内 部 会 触发 计时 器 。 

(2) updateUsingFeed: 是 HPNewsFeedUpdateTask 对 象 周期 性 调用 的 回调 方法 。 

(3) dealloc Jj iA O 负责 调用 任务 对 象 的 shutdown 方法 ， 其 内 部 会 销毁 计时 器 。 注 意 ， 
dealloc 在 此 处 是 明确 可 用 的 ， 因 为 该 对 象 没有 被 其 他 的 地 方 所 引用 。 


当 使 用 NSTimer 和 NSThread 时 ， 总 是 应 该 通过 间接 的 层 实现 明确 的 销毁 过 
程 。 这 个 间接 层 应 使 用 弱 引 用 ， 从 而 保证 所 拥有 的 对 象 能 够 在 停止 使 用 后 执 
行销 毁 动作 。 
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2.10.3 ”观察 者 

与 使 用 委托 和 订阅 复杂 数据 变化 的 回调 不 同 ， 系 统 提 供 了 两 种 内 置 的 可 用 选择 ， 以 便 监听 
变化 。 之 所 以 说 它们 是 内 置 的 ， 主 要 是 因为 我 们 无 需 编写 任何 自 定 义 的 代码 来 跟踪 观察 
者 一 一 运行 时 对 管理 观察 者 提供 了 支持 。 这 类 技术 包括 : 

。 键 - 值 观察 

。 通知 中 心 

1. 键 - 值 观 察 

Objective-C 允许 用 addObserver:forKeyPath:options:context: 方法 在 任何 NSObject 子 类 的 
对 和 象 上 添加 观察 者 。 观 察 者 会 通过 observeValueForKeyPath:ofObject:change:context: Fy 
i5 $3 XI, removeObserver:forKeyPath:context: 方法 用 于 解除 注册 或 移 除 观察 者 。 这 就 
是 众所周知 的 键 — 值 观 察 。 

这 是 一 个 极为 有 用 的 特性 ， 尤 其 是 在 以 调试 为 目的 跟踪 某 些 共享 于 应 用 多 个 部 分 (如 用 户 
接口 、 业 务 逻辑 、 持 久 化 以 及 网 络 ) 的 对 象 时 。 

举 一 个 相关 的 示例 ， 类 似 的 对 象 可 能 是 某 个 自 定义 类 ， 它 持 有 着 应 用 当前 的 状态 细 市 。 例 
如 ， 标 识 用 户 是 否 登录 、 已 登录 的 用 户 信息 、 电 子 商 务 应 用 中 购物 车 的 内 容 、 或 者 用 户 在 
信息 应 用 内 最 新 消息 的 接收 入。 为 了 方便 调试 ， 你 可 能 会 为 这 个 对 象 添 加 一 个 观察 者 ， 以 
跟踪 所 有 的 修改 或 更 新 。 

键 - 值 观察 在 双 癌 数据 绑 定 中 也 非常 有 用 。 视 图 可 以 关联 委托 来 响应 那些 会 导致 模型 更 新 
的 用 户 交 互 。 键 - 值 观察 可 以 用 于 反 向 的 绑 定 ， 以 便 在 模型 发 生变 化 时 更 新 UI。 


以 下 内 容 摘自 官方 文档 (https://sites.google.com/site/appleiotemp//11Bd01C)。 










































































键 - 值 观察 方法 addObserver:forKeyPath:options:context: A 4 Zi d$ XL ER XT E. 
被 观察 对 象 及 上 下 文 对 象 的 强 引 用 。 如 有 必要 ， 你 需要 自行 维护 对 它们 的 强 引 用 。 


这 意味 着 观察 者 需要 有 足够 长 的 生命 周期 才能 够 持续 地 监控 变化 。 你 需要 额外 关注 观察 者 
的 生命 周期 ,而 且 要 持续 到 所 观察 的 内 存 被 废弃 之 后 。 


例 2-32 用 和 集中 式 的 ObserverManager 2K 3e sz II HE — {A WL 2$, ObserverManager 类 会 返 
回 一 个 与 持 有 者 相关 的 ObserverObserveeHandle 实例 。 当 观察 启动 程序 (示例 中 为 
视图 控制 器 ) 需要 观察 keyPath， 它 会 调用 addObserverToObject:forKey: 方法 并 储存 
ObserverObserveeHandle， 从 而 实现 与 视图 控制 器 同时 销毁 。handle 会 在 其 销毁 期 间 移 除 
观察 者 。 

这 里 尝试 解决 的 问题 本 质 上 与 NSTimer 的 例子 相似 。 只 是 计时 器 的 场景 需要 建立 一 个 弱 引 
用 ， 而 这 里 如 果 没 有 正确 处 理 ， 则 会 导致 观察 者 永久 被 销毁 。 


例 2-32 键 - 值 观察 


(interface ObserverObserveeHandle 










































































(property (nonatomic, strong) MyObserver *observer; 
(property (nonatomic, strong) NSObject *obj; 
(property (nonatomic, copy) NSString *keyPath; 
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-(id)initWithObserver: (MyObserver *)observer 
target: (NSObject *)obj 
keyPath: (NSString *)keyPath; 


@end 
(implementation ObserverObserveeHandle 


-(id)initWithObserver:(MyObserver *)observer 
target:(NSObject *)obj 
keyPath:(NSString *)keyPath { 
// 删 除 以 便 更 加 简洁 
} 


-(void)removeObserver { 

[self.obj removeObserver:self forKeyPath:self.keyPath context:nil]; 
self.obj = nil; 
} 


-(void)dealloc { 
[self removeObserver]; 


} 
@end 


@interface ObserverManager 
// 删 除 以 便 更 加 简洁 
Qend 


(implementation ObserverManager 
NSMutableArray *observers; 


*(ObserverObserveeHandle)addObserverToObject:(NSObject *)obj 
forKey:(NSString *)keyPath { 

MyObserver *observer = [[MyObserver alloc] init]; 

[obj addObserver:observer forKeyPath:keyPath 
options:(NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld) 
context:NULL]; 


ObserverObserveeHandle *details - [[ObserverObserveeHandle alloc] 
initWithObserver:observer target:obj keyPath:keyPath]; 
[observers addObject:details]; 


return details; 


} 
@interface SomeViewController 


@property (nonatomic, strong) IBOutlet UILabel *resultLabel; 
@property (nonatomic, strong) ObserverObserveeHandle *resultLabelMonitor ; 


@end 


(implementation SomeViewController 
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-(void)viewDidLoad { 
self.resultLabelMonitor = [ObserverManager 
addObserverToObject:self.nameTextField 

forKey:Q" text" ]; 
} 


@end 





当 你 为 目标 对 象 添 加 键 - 值 观察 者 时 ， 目 标 对 象 的 生命 周期 至 少 应 该 和 观察 
者 一 样 长 ， 因 为 只 有 这 样 才 有 可 能 从 目标 对 象 移 除 观 察 者 。 这 可 能 会 导致 目 
标 对 象 的 生命 周期 比 预 期 要 长 ， 也 是 你 需要 额外 小 心 的 地 方 。 














例 2-32 似乎 提供 了 一 个 优秀 的 解决 方案 ， 通 过 风格 良好 且 不 会 出 错 的 代码 ， 该 示例 解除 了 
执行 清理 带 来 的 负担 。 然 而 ， 这 里 仍然 存在 问题 ， 就 是 观察 者 的 所 有 通知 都 会 执行 相同 的 
代码 片段 (如 Observer 类 定义 的 代码 ) 。 


如 何 解 决 这 个 问题 呢 ? 这 很 值得 思 芳 。 提 示 : 使 用 块 方法 。 在 块 方法 内 ， 如 果 需 要 调用 使 
用 self 的 方法 ， 别 忘 了 在 将 它 用 于 块 方法 内 部 的 使 用 前 先 创建 对 self 的 弱 引 用 。 


于 是 ， 注 册 观 察 者 的 代码 得 到 了 更 新 ， 如 例 2-33 所 示 。 
例 2-33 使 用 块 方法 的 键 - 值 观察 者 


@implementation SomeViewController 





























-(void)viewDidLoad 


. weak typeof(self) weakSelf = self; 

self.resultLabelMonitor - [ObserverManager 
addObserverToObject: self.nameTextField 
forKey:Q" text" block:^(NSDictionary *changes) { 


typeof(self) sSelf - weakSelf; 
if(sself) { 
NSLog(@"Text changed to %@", 
[changes objectForKey:NSKeyValueChangeNewKey ] ) ; 


// 需 要 时 使 用 sSelf 
sSelf.resultLabel.text = Q"Name changed"; 


H; 


Qend 
2. 通知 中 心 
另 一 个 方案 是 使 用 通知 中 心 。 一 个 对 象 可 以 注册 为 通知 中 心 (NSNotificationCenter 对 象 ) 
的 观察 者 ， 并 接收 NSNotification 对 象 。 与 键 — 值 观 察 者 相似 ， 通 知 中 心 不 会 对 观察 者 持 
有 强 引 用 。 这 意味 着 开发 人 员 得 到 了 解放 ， 无 需 为 观察 者 的 析 构 过 早 或 过 晚 而 操心 。 
有 具体 的 解决 模式 与 我 们 在 前 一 节 讨 论 的 内 容 相似 。 
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2.10.4 返回 错误 

当 用 某 个 方法 接收 NSError ** 参数 ， 并 在 发 生 错 误 时 填充 错误 变量 ， 则 必须 使 用 
. autoreleasing 限定 符 。 使 用 这 一 范式 的 最 常见 场景 是 需要 处 理 输入 并 返回 可 能 包含 错误 
的 值 的 时 候 。 

例 2-34 展示 了 典型 情况 的 方法 签名 。 


例 2-34 返回 错误 
-(Matrix *)transposeMatrix:(Matrix *)matrix error:(NSError * 
error 


























autoreleasing *) 


// 处 理 
// 如 果 发 生 错 误 


*error = [[NSError alloc] initWithDomain:Q"transpose" code:123 userInfo:nil]; 





} 
需要 关注 这 一 语法 。 关 键 字 _autoreleasing 要 塞 到 两 个 星 号 之 间 。 要 牢记 这 一 点 


NSError* — autoreleasing *error; 
你 可 能 已 经 注意 到 了 ， 变 量 和 属性 的 限定 符 有 重要 的 作用 ， 可 以 帮助 管理 和 精确 控制 对 象 
的 生命 周期 确保 它们 既 不 会 大 长 也 不 会 过 短 。 每 当 存 有 疑问 时 ， 你 都 应 该 翻 回去 看 看 画 
板 、 回 顾 一 下 基础 知识 ， 并 合理 地 定义 属性 和 变量 。 有 时 你 需要 用 强 引 用 修饰 创建 的 属性 


并 延长 其 生命 周期 ， 而 有 时 你 却 需 要 使 用 弱 引 用 并 确保 获取 适当 的 内 存 使 用 ， 以 避免 内 存 
itii. 


2.11 PW: 
许多 场景 都 需要 使 用 id 类 型 。 在 Cocoa framework 中 见 到 此 类 型 也 绝 非 罕见 。 例 如 ， 在 
Xcode 生成 的 代码 中 ，IBAction 方法 使 用 了 id 类 型 的 参数 来 表示 sender, 

另 一 个 场景 是 使 用 NSArray 中 的 对 象 。“ 思考 例 2-35 中 的 代码 。 


例 2-35 使 用 NSArray 中 的 对 象 


@interface HPDataListViewController 
: UITableViewController <UITableViewDataSource, UITableViewDelegate> 
























































(property (nonatomic, copy) NSArray *input; 
Qend 
(implementation HPDataListViewController 


-(void)tableView:(UITableView *)tableView 





B 
Hr 
ON 





: iOS 9 introduces lightweight generics for Objective-C collections for interoperability with Swift. See iOS Developer 
Library, “Lightweight Generics”. (https://developer.apple.com/library/content/documentation/Swift/Conceptual/ 
BuildingCocoaA pps/Interacting WithObjective-CAPIs.html#//apple_ref/doc/uid/TP40014216-CH4-ID35) 
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didSelectRowAtIndexPath:(NSIndexPath *)indexPath 


{ 


NSUInteger value = [self.input objectAtIndex: indexPath. row].someProperty; 
// 继 续 


Qend 


在 tableView:didSelectRowAtIndexPath: 方法 中 ， 我 们 将 某 种 类 型 (假设 为 CLassX， 其 中 
包含 一 个 属性 someProperty) 的 对 象 组 成 的 数组 作为 输入 。 

这 上段 代码 看 起 来 很 好 ， 而 且 如 果 尝 试 运行 ， 你 也 很 可 能 会 得 到 正确 的 结果 。 我 们 知道 不 同 
下 标的 对 象 一 直 都 会 响应 someProperty 的 选择 器 ， 所 以 这 段 代 码 会 按照 预期 工作 。 但 如 果 
某 个 对 象 没有 响应 这 个 选择 器 ， 那 么 就 会 导致 应 用 崩 江 。 
假设 编译 器 并 不 需要 知道 类 型 信息 ， 因 为 运行 时 知道 应 该 调用 哪个 对 象 和 方法 。 但 事实 
是 ， 编 译 器 确实 需要 了 解 细 节 情 况 一 一 具体 来 说 ， 它 需要 知道 所 有 参数 的 大 小 和 返回 结果 
的 类 型 ， 这 样 才 能 生成 正确 的 指令 ， 正 确 地 在 栈 上 压 入 和 弹出 数据 。 举 例 来 说 ， 如 果 方 法 
有 两 个 int 类 型 的 参数 ， 则 需要 在 栈 上 压 入 8 FH. 

通常 情况 下 ， 我 们 无 需 关注 这 些 细 市 。 在 获取 参数 信息 时 ， 编 译 器 会 根据 所 调用 的 方法 名 
称 遍 历 查 找 导 入 的 头 文件 以 匹配 方法 名 ， 然 后 通过 找到 的 第 一 个 匹配 的 方法 获取 参数 长 度 。 
这 个 方案 好 在 适用 于 绝 大 多 数 情况 。 但 如 果 多 个 类 有 完全 相同 的 方法 签名 (如 名 称 和 参 
数 )， 那 么 将 无 法 正常 工作 。 

考虑 以 下 这 个 场景 ， 在 编译 时 ， 编 译 器 没有 聚焦 在 ClassX 上 ， 假 设 是 在 Classy 对 象 上 。 
那么 方法 可 能 不 会 返回 NSUInteger， 而 可 能 会 返回 NSInteger， 甚 至 返回 NSString。 在 
另外 一 个 场景 中 ， 我 们 原本 预期 返回 NSUInteger ， 而 实际 返回 了 某 种 需要 我 们 对 其 进行 
invalidate my cleanup 的 对 象 (如 CGColor 或 CGContext)， 这 最 终 会 导致 内 存 泄 漏 。 


问题 的 解决 方案 

为 何 会 发 生 类 型 匹配 错误 呢 ? 编译 器 怎 
决 对 象 的 消息 发 送 。 编 译 器 负责 生成 精 
的 值 )。 

好 在 编译 器 解决 类 型 匹配 错误 的 问题 并 不 困难 。 解 决 方案 分 为 两 个 部 分 。 


首先 ， 我 们 必须 对 编译 器 进行 配置 ， 当 在 id 对 象 上 发 现 多 个 选择 器 匹配 时 ， 编 译 器 要 报 
告 错误 。 这 是 由 Strict Selector Matching 选项 控制 的 ， 默 认为 关闭 状态 。 与 之 对 应 的 编译 器 
参数 是 -Wstrict-selector-match。 当 编译 器 发 现 两 个 选择 器 有 不 同 的 参数 或 返回 类 型 时 ， 
将 其 打开 可 以 发 出 警告 。 


2-14 展示 了 Xcode 中 的 工程 设置 。 
与 此 选项 有 关 的 一 些 问 题 : 
。 内 置 的 框架 会 产生 许多 警告 ， 尽 管 绝 大 多 数 警 告 并 不 会 带 来 任何 麻烦 ， 
























































么 会 如 此 不 成 熟 ” 它 进行 了 艰 苗 卓绝 的 工作 来 解 
确 的 指令 〈 例 如， 传递 给 objc_msgSend 方法 正确 





























二 
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。 如 有 果 是 处 理 类 而 不 是 对 象 ， 那 么 你 仍然 无 法 发 现 这 种 问题 ; 
。 如 果 导 入 的 头 文件 没有 正确 的 定义 ， 那 么 这 个 选项 也 不 会 有 任何 帮助 。 















































im | < > | [f HighPerformance <A 
7 General Capabilities Info Build Settings Build Phases Build Rules 
PROJECT Basic (AI) | (Combined) Levels | + (Qr Strict Selector ©) 
ff HighPerformance 
TARGETS Y Apple LLVM 5.1 - Warnings - Objective C 
HighPerformanc Setting 贺 HighPerformance 
(“HighPerformanceTests Strict Selector Matching No * 














图 2-14: 在 XCode 中 设置 严格 的 选择 器 匹配 

这 将 我 们 引入 了 解决 方案 的 第 二 部 分 : 为 编译 器 提供 足够 多 的 信息 ， 以 供 其 生成 基于 正确 
类 型 的 信息 。 你 可 以 使 用 强 类 型 〈 例 中 所 使 用 的 是 classX)。 例 2-36 展示 了 对 代码 的 改动 。 
例 2-36 使 用 强 类 型 


-(void)tableView:(UITableView *)tableView 
didSelectRowAtIndexPath:(NSIndexPath *)indexPath 





{ 
ClassX *item = (ClassX *) [self.input objectAtIndex:indexPath.row]; 
NSUInteger value = item.someProperty; 
// 继 续 

} 


简 而 言 之 ， 在 使 用 常规 命名 的 方法 时 ， 应 避免 使 用 id。 尽量 使 用 具体 的 类 取而代之 。 


A Eu 
212 ”对 象 寿命 与 泄漏 
对 象 在 内 存 中 活动 的 时 间 越 长 ， 内 存 不 能 被 清理 的 可 能 性 就 越 大 。 所 以 应 当 尽 可 能 地 避免 
出 现 长 寿命 的 对 象 。 当 然 ， 你 需要 保留 代码 中 关键 操作 对 象 的 引用 ， 为 的 是 不 必 每 次 都 浪 
费时 间 来 创建 它们 。 尽 量 在 使 用 这 些 对 象 时 完成 对 它们 的 引用 。 
长 寿命 对 象 的 常见 形式 是 单 例 。 日 志 器 是 典型 的 例子 一 一 只 创建 一 次 ， 从 不 销毁 。 我 们 会 
在 下 一 市 深入 讨论 此 类 情况 。 
另 一 个 方案 是 使 用 全 局 变量 。 全 局 变量 在 程序 开发 中 是 可 怕 的 东西 。 
要 想 合理 地 使 用 全 局 变量 ， 必 须 满足 以 下 条 件 : 
。 没有 被 其 他 对 象 所 持 有 ， 
不 是 常量 ， 
。 整个 应 用 中 只 有 一 个 ， 而 不 是 每 个 组 件 一 个 。 
如 果 某 个 变量 不 符合 这 些 要 求 ， 那 么 它 不 应 该 被 用 作 全 局 变量 。 
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复杂 的 对 象 图 使 得 回收 内 存 的 机 会 变 得 更 少 ， 同 时 增加 了 应 用 因 内 存 耗 尽 而 天福 的 风险 。 
如 果 主 线程 总 是 被 迫 等 待 子 线程 的 操作 (如 网 络 或 数据 库存 取 )， 那 么 应 用 的 响应 性 能 会 
变 得 很 差 。 


2.13 单 例 
单 例 模式 是 限制 一 个 类 只 初始 化 一 个 对 象 的 一 种 设计 模式 。 在 实践 中 ， 初 始 化 常常 在 应 用 
启动 不 久 后 执行 ， 而 且 这 些 对 象 不 会 被 销毁 。 

让 一 个 对 象 有 着 与 应 用 一 样 长 的 生命 周期 可 不 是 什么 好 主意 。 如 果 这 个 对 象 是 其 他 对 象 的 

源头 (如 一 个 服务 定位 器 ) ， 若 定位 器 的 实现 不 正确 则 有 可 能 造成 内 存 风 险 。 

训 无 疑问 ， 单 例 是 必要 的 。 但 单 例 的 实现 对 其 使 用 方式 有 重要 影响 。 

在 充分 讨论 单 例 引 入 的 问题 之 前 ， 我 们 不 妨 先 更 好 地 理解 单 例 ， 了 解 一 下 为 什么 确实 需要 

使 用 单 例 。 

和 例 极为 有 用 ， 尤 其 是 在 某 个 系统 确定 只 需要 一 个 对 象 实例 时 。 应 该 在 以 下 情形 中 使 用 

B fh : 

。 队列 操作 (如 日 志和 埋 点 ) 

。 访问 共享 资源 (如 缓存 ) 

。 资源 池 (如 线程 池 或 连接 池 ) 

一 旦 创建 ， 单 例会 一 直 存 活 到 应 用 关闭 。 日 志 器 、 埋 点 服务 以 及 缓存 都 是 使 用 单 例 的 合理 

场景 。 

更 重要 的 是 ， 单 例 通 常会 在 应 用 启动 时 进行 初始 化 ， 打 算 使 用 单 例 的 组 件 需 要 等 它们 准备 

得 当 。 这 会 增加 应 用 的 启动 时 间 。 

有 办 法 解决 吗 ? 现 在 还 没有 。 如 果 越 来 越 多 地 使 用 一 些 现 成 的 第 三 方 组 件 ， 内 存 窘迫 的 情 

况 会 不 断 加 剧 ， 尤 其 是 在 没有 它们 的 源码 的 情况 下 。 

你 可 以 使 用 以 下 的 指导 原则 。 

。 尽 可 能 地 避免 使 用 单 例 。 

。 识别 需要 内 存 的 部 分 ， 如 用 于 埋 点 的 内 存 缓冲 区 (在 尚未 将 数据 同步 到 服务 器 前 使 用 )。 
寻求 减少 内 存 的 方法 。 注 意 ， 你 需要 将 减少 内 存 与 其 他 事情 做 权衡 。 减 小 缓冲 区 意味 着 
更 多 的 服务 器 通信 。 

。 尽量 避免 对 象 级 的 属性 ， 因 为 它们 会 与 对 象 共存 它 。 尽 量 使 用 本 地 变量 。 
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依赖 注入 
完全 避免 单 例 可 能 并 不 容易 ， 但 至 少 可 以 尽量 避免 直接 使 用 它们 。 应 避免 编写 例 2-37 
这 样 的 代码 。 


例 2-37 不 合理 地 使 用 依赖 


-(void)someMethod { 
XXSomeClass *obj = [XXSomeClass sharedInstance]; @ 
NSString *someValue = [obj operation:@"some parameter"]; 
// 继 续 

} 


@ someMethod 为 operation 方法 使 用 了 XXSomeClass, 
在 例 2-37 中 ，someMethod 依赖 了 外 部 的 类 XXSomeClass， 不 受 应 用 设置 的 控制 管理 。 
这 一 切 工作 良好 ， 但 存在 一 些 潜在 的 问题 。 


。 Je X X XXSomeClass 需要 某 些 初始 化 ，someMethod 假设 它 已 经 完成 了 初始 化 。 然 
而 ， 事 实 上 XXSoneClass 并 不 为 上 游 使 用 someMethod 的 方法 所 知 ， 这 可 能 会 导致 
XXSomeClass 尚未 初始 化 。 

。 如 果 类 XXSomeClass 持 有 了 一 些 资源 ， 那 么 它 会 持续 持 有 ， 哪 怕 sharedInstance 后 
续 不 会 再 被 调用 。 

要 想 避 免 此 类 陷阱 ， 请 使 用 依赖 注入 。 依 赖 注入 本 质 上 是 在 需要 时 传递 依赖 。 取 决 于 

依赖 的 范围 ， 依 赖 注 入 可 以 通过 自 定 义 的 初始 化 器 或 调用 方法 实现 注入 。 

如 果 被 依赖 的 对 象 在 类 中 的 多 个 地 方 被 使 用 ， 那 么 执行 注入 的 最 佳 地 方 就 是 自 定 义 的 

初始 化 器 。 如 果 依 赖 的 对 象 只 在 少量 几 个 操作 中 ， 且 能 为 这 些 操作 提供 不 同 的 实例 ， 

那么 应 该 对 每 个 方法 分 别 注入 。 

例 2-38 中 的 更 新 代码 演示 了 依赖 注入 的 两 种 方案 : 使 用 了 Typhoon (http://typhoon- 

framework.org) 和 Objection (http://objection-framework.org) ， 二 者 是 常见 且 开 发 活跃 

的 依赖 注入 框架 。 


例 2-38 使 用 依赖 注入 的 更 新 代码 


@interface MyClass 
-(instancetype)initWithSomeClass:(XXSomeClass *)someClass; @ 


- (void) someMethod; 
-(void)anotherMethodWithAnotherClass:(AnotherClass *)anotherClass; €) 


Qend 
@interface MyClass () 


@property (nonatomic, strong) XXSomeClass *someClass; 
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Qend 
(implementation MyClass 


-(instancetype)initWithSomeClass:(XXSomeClass *)someClass { 
if(self = [super init]) { 
self.someClass - someClass; 


j 


return self; 


j 


-(void)someMethod { © 
NSString *someValue = [self.someClass operation:@"some parameter" ]; 
// 继 续 

} 


-(void)anotherMethodWithAnotherClass:(AnotherClass *)anotherClass { @ 
NSString *someValue = [self.someClass operation:@"some parameter"]; 
NSString *anotherValue = [anotherClass anotherOp:@"another parameter"]; 

i // 继 续 


(gend 


Q 自 定义 需要 传 入 XXsoneClass 对 象 的 初始 化 器 。 

@ anotherMethodWithAnotherClass 使 用 了 AnotherClass 对 象 ， 并 需要 它 以 参数 形式 
传 入 。 

Q) someMethod 现在 通 someClass 属性 调用 operation 方法 。 

@ anotherMethodWithAnotherClass 现在 可 以 使 用 someClass 和 AnotherClass 类 型 的 另 
一 个 对 象 来 完成 任务 。 











2.14 找到 神秘 的 持 有 者 


就 算 类 设计 精良 ， 对 象 被 良好 持 有 ， 是 否 会 发 生 内 存 泄漏 还 是 不 确定 的 。 如 果 发 生 内 存 泄 
漏 ， 获 取 引 用 图 是 一 个 不 错 的 解决 办 法 。 那 么 问题 来 了 ， 是 否 能 找 出 一 个 对 象 上 的 全 部 引 
用 呢 ? 


答案 位 于 ARC 之 前 的 方法 retain。 我 们 需要 计算 这 个 方法 的 调用 次 数 。ARC 禁止 覆盖 或 
调用 这 个 方法 ， 但 你 可 以 暂时 在 工程 中 禁用 ARC (详情 参见 图 2-15)。 


















































mance 
General Capabilities Info Build Settings Build Phases Build Rules 


Basic | Levels | + (Q- clang, enable 








V Apple LLVM 5.1 - Language - Modules 
Setting Æ HighPerformance 
Enable Modules (C and Objective-C) Yes $ 


la 


V Apple LLVM 5.1 - Language - Objective C 
Setting Eg HighPerformance 


Objective-C Automatic Reference Counting Yes Y 








图 2-15; 在 工程 中 禁用 ARC 

然后 将 例 2-39 中 的 代码 添加 到 所 有 的 自 定义 类 中 。 这 段 代 码 不 仅 会 记录 对 retain 方法 的 调 
用 情况 ， 还 会 将 调用 栈 打印 出 来 。 因 此 ， 除 了 调用 次 数 ， 你 还 可 以 得 到 精确 的 调用 明细 。 
例 2-39 使 用 retain 获取 引用 计数 


#if ! has feature(objc arc) 
-(id) retain 








DDLogInfo(@"%s %@", __PRETTY_FUNCTION__, [NSThread callStackSymbols]); 
return [super retain]; 


} 
Zendif 
2.15 ”最 佳 实践 
通过 遵循 这 些 最 佳 实践 ， 你 将 很 大 程度 上 避免 许多 麻烦 ， 如 内 存 汽 漏 、 循 环 引 用 和 较 大 内 


存 消耗 。( 你 可 以 将 这 一 部 分 打印 出 来 ， 挂 在 工 位 上 ， 以 便 快 速 查看 。) 

。 避免 大 量 的 单 例 。 具 体 来 说 ， 不 要 出 现 上 沉 对 和 象 〈 如 职责 特别 多 或 状态 信息 特别 多 的 对 
象 ) 。 这 是 一 个 反 模 式 ， 指 代 一 种 常见 解决 方案 的 设计 模式 ， 但 很 快 产 生 了 不 恨 效果 。 
日 志 器 、 埋 点 服务 和 任务 队列 这 样 的 辅助 单 例 都 是 很 不 错 的 ， 但 全 局 状态 对 象 不 可 取 。 

。 对 子 对 象 使 用 _strong。 

。 对 父 对 象 使 用 weak, 

。 对 使 引用 图 闭合 的 对 象 (如 委托 ) 使 用 _weak。 

。 对 数值 属性 (NSInteger, SEL, CGFloat 等 ) 而 言 ， 使 用 assign 限定 符 。 

。 对 于 块 属性 ， 使 用 copy 限定 符 。 
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。 当 声 明 使 用 NSError ** 参数 的 方法 时 ， 需 要 使 用 _autoreteasing， 并 要 注意 用 正确 的 
语法 : NSError * | autoreleasing *, 

。 避免 在 块 内 直接 引用 外 部 的 变量 。 在 块 外 面 将 它们 weakify， 并 在 块 内 再 将 它们 
strongify。 参见 libextobjc JÆ (https://github.com/jspahrsummers/libextobjc) 来 了 解 
@weakify 和 @strongify, 

。 进行 必要 清理 时 遵循 以 下 准则 : 

4 SIN at 
4 移 除 观察 者 (具体 来 说 ， 移 除 对 通知 的 注册 ) 
e 解除 回调 (具体 来 说 ， 将 强 引 用 的 委托 设置 为 nil) 


2.16 生产 环境 的 内 存 使 用 情况 


注意 ， 无 论 你 对 Xcode 做 了 什么 设置 ， 它 只 会 对 调试 应 用 起 作用 。 当 一 款 应 用 还 没有 真正 
流行 起 来 ， 只 有 数 万 而 不 是 数 百 万 用 户 时 ， 通 常 无 需 关注 这 些 设置 所 导致 的 额外 变化 。 

要 想 在 不 同 的 情境 中 分 析 应 用 ， 你 可 以 使 用 埋 点 。 定 期 将 应 用 的 信息 发 送 给 服务 器 ， 发 
送 的 信息 包括 内 存 的 消耗 ， 尤 其 是 如 果 内 存 超出 装 值 时 配合 一 些 辅助 定位 信息 是 很 不 错 
的 选择 。 

例如 ， 如 果 内 存 消耗 超过 了 40MB, ， 你 可 能 需要 向 服务 器 发 送 一 些 用 户 所 在 页 面 的 细节 
信息 以 及 所 执行 的 关键 操作 。 男 一 个 选择 是 跟踪 内 存 的 消耗 ， 以 一 定 的 间隔 在 本 地 记录 
日 志 ， 然 后 再 将 数据 上 报 给 服务 器 。 你 可 以 用 例 2-40 中 的 代码 来 找到 已 经 使 用 和 可 用 的 
内 存 。 





















































在 出 现 低 内 存 警 告 时 对 应 用 进行 埋 点 ， 包 括 内 存 的 使 用 及 统计 信息 ， 并 在 应 
用 重新 运行 时 将 这 些 信息 上 报 给 服务 器 。 使 用 这 些 信 息 来 识别 出 应 用 发 生 内 
存 溢出 的 常见 场景 和 边缘 情况 。 

















例 2-40 跟踪 可 用 和 已 用 的 内 存 


//HPMemoryAnalyzer.m 
#import <mach/mach.h> 


vm_size_t getUsedMemory() { 
task_basic_info_data_t info; 
mach_msg_type_number_t size = sizeof(info); 
kern return t kerr = task info(mach task self(), TASK BASIC INFO, 
(task info t) &info, &size); 


if(kerr == KERN SUCCESS) { 
return info.resident size; 
} else { 
return 0; 





j 


vm size t getFreeMemory() { 

mach port t host - mach host self(); 

mach msg type number t size - sizeof(vm statistics data t) / sizeof(integer t); 
vm size t pagesize; 

vm statistics data t vmstat; 


host page size(host, &pagesize); 
host statistics(host, HOST VM INFO, (host info t) &vmstat, &size); 


return vmstat.free count * pagesize; 





使 用 Instruments 进行 内 存 分 析 


你 可 以 使 用 Xcode Instruments 工具 对 应 用 的 内 存 使 用 进行 分 析 。 我 们 将 在 11.2 节 深 入 
介绍 Instruments 工具 。 我 们 对 其 中 的 Allocations 和 Leaks 尤其 感 兴趣 。 
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现在 你 已 经 深入 了 解 了 iOS 运行 时 是 如 何 管理 内 存 的 ， 并 知道 了 避免 发 生 循环 引用 (AE 
泄漏 的 最 大 单一 来 源 ) 的 基本 规则 。 现 在 你 可 以 减少 内 存 的 消耗 、 降 低 平均 和 峰值 内 存 需 
求 了 。 
你 可 以 使 用 僵尸 对 象 来 跟踪 多 次 释放 的 对 象 (造成 应 用 崩溃 的 最 主要 原因 之 一 )。 

本 章 还 提供 了 用 于 跟踪 内 存 使 用 情况 的 代码 ， 你 不 仅 可 以 在 实验 室 的 测试 环境 中 运行 这 些 
代码 ， 还 可 以 在 生产 环境 中 运行 。 
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如 今 的 移动 设备 早已 无 处 不 在 ， 待 机 时 间 成 为 影响 设备 最 终 销量 的 一 个 重要 因素 。 类 似 
地 ， 电 量 消耗 也 是 决定 其 个 应 用 是 否 会 被 安装 的 重要 因素 之 一 。 

设备 中 的 每 个 硬件 模块 都 会 消耗 电量 。 电 量 的 最 大 消费 者 是 CPU， 但 这 只 是 系统 的 一 个 方 
面 。 一 个 编写 良好 的 应 用 需要 谨慎 地 使 用 电能 。 用 户 往 往 会 删除 耗 电量 大 的 应 用 。 

BR CPU 外 ， 耗 电量 高 、 值 得 关注 的 硬件 模块 还 包括 : 网 络 硬 件 、 监 牙 、GPS、 麦 克 风 、 加 
速 计 、 摄 像 头 、 扬 声 器 和 屏幕 。 

本 章 重 点 关注 消耗 电量 的 关键 领域 ， 以 及 如 何 降低 电量 的 消耗 。 我 们 将 学 习 如 何 编 写 能 判 
断 电 池 的 剩余 电量 及 充电 状态 的 应 用 ， 还 将 讨论 如 何在 iOS 应 用 中 分 析 电 源 、CPU 和 资源 
的 使 用 。 


3.1 CPU 
不 论 用 户 是 否 正在 直接 使 用 ，CPU 都 是 应 用 所 使 用 的 主要 硬件 。 在 后 台 操 作 和 处 理 推送 通 
知 时 ， 应 用 仍 会 消耗 CPU 资源 。 


iPhone (5, 5S 和 6) 和 iPad (3、4 和 Air) 的 处 理 器 是 双核 或 三 核 的 。 你 可 以 在 表 3-1 中 查 
完整 的 列表 。Geekbench 的 打分 反映 了 这 些 常见 主流 OS 设备 处 理 器 的 相对 计算 速度 。 
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表 3-1: iOS 设 备 与 处 理 器 





设备 处 理 器 ”核心 数 ”地 址 长 度 CPU 时 钟 单 核 Geekbench ”多 核 Geekbench' 
iPhone 5 A6 2 32 bit 1.3 GHz 569 950 
iPhone 5S AT 2 64 bit 1.3~1.4 GHz 1400 2524 
iPhone 5C A6 2 32 bit 1.3 GHz 689 1243 
iPhone 6 A8 2 64 bit 1.4 Ghz 1621 2899 
iPhone 6 Plus A8 2 64 bit 1.4 Ghz 1619 2902 
iPhone 6S A9 2 64 bit 1.8 Ghz 2487 4327 
iPhone 6S Plus A9 2 64 bit 1.8 Ghz 2478 4330 
iPad 3 ASX 2 32 bit 1 Ghz 261 495 
iPad 4 A6X 2 32 bit 1.4 Ghz 781 1422 
iPad Air AT 2 64 bit 1.4 Ghz 1462 2636 
iPad Air 2 AS8X 3 64 bit 1.5 Ghz 1815 4502 


应 用 计算 得 越 多 ， 消 耗 的 电量 就 越 多 。 在 完成 相同 的 基本 操作 时 ， 老 一 代 的 设备 会 消耗 更 
多 的 电量 。 计 算 量 的 消耗 取决 于 不 同 的 因素 。 





对 数据 的 处 理 (例如 ， 对 文本 进行 格式 化 )。 

待 处 理 的 数据 大 小 一 一 更 大 的 显示 屏 允 许 软 件 在 单个 视图 中 展示 更 多 的 信息 ， 但 这 也 意 
味 着 要 处 理 更 多 的 数据 。 

处 理 数 据 的 算法 和 数据 结构 。 

执行 更 新 的 次 数 ， 尤 其 是 在 数据 更 新 后 ， 触 发 应 用 的 状态 或 UI 进行 更 新 (应 用 收 到 的 
推送 通知 也 会 导致 数据 更 新 ， 如 果 此 时 用 户 正 在 使 用 应 用 ， 你 还 需要 更 新 UI), 


















































没有 单一 规则 可 以 减少 设备 中 的 执行 次 数 。 很 多 规则 都 取决 于 操作 的 本 质 。 以 下 是 一 些 可 
以 在 应 用 中 投入 使 用 的 最 佳 实践 。 








针对 不 同 的 情况 选择 优化 的 算法 

例如 ， 当 你 在 排序 时 ， 如 有 果 列 表 少 于 43 个 实例 ， 则 播 和 排序 优 于 归并 排序 ， 但 实例 多 
于 286 个 时 ， 应 当 使 用 快速 排序 。 要 优先 使 用 双 枢 轴 快 速 排序 而 不 是 传统 的 单 枢 轴 快速 
排序 。 


如 果 应 用 从 服务 器 接收 数据 ， 尽 量 减少 需要 在 客户 端 进行 的 处 理 
例如 ， 如 果 一 段 文字 需要 在 客户 端 进行 泻 染 ， 尽 可 能 在 服务 器 将 数据 清理 干净 。 


我 曾经 做 过 一 个 项 目 ， 因 为 服务 器 的 实现 主要 用 于 服务 桌面 用 户 ， 所 以 返回 的 文本 中 包 
含 HTML 标签 。 清 理 HTML 标签 的 工作 并 没有 放 在 客户 端 进行 ， 而 是 放 在 了 服务 器 端 
实现 ， 从 而 减少 了 设备 上 的 计算 过 程 ， 降 低 了 处 理 时 间 。 


我 们 在 另 一 个 项 目 中 意识 到 ， 如 果 每 次 用 户 开启 应 用 的 数据 差异 较 大 ， 需 要 同步 的 记录 
开销 也 是 相当 高 的 。 我 们 并 没有 用 传统 的 方式 ， 通 过 服务 器 下 发 更 新 的 增 量 数据 ， 而 是 
将 其 设置 成 发 送 一 个 二 进 制 的 数据 库 ， 用 于 替换 设备 上 已 有 的 数据 库 。 这 不 仅 确保 了 对 
网 络 的 优化 使 用 ， 还 节省 了 在 本 地 设备 上 合并 数据 所 需 的 计算 。 
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E 1; https://browser.primatelabs.com/ios-benchmarks 








。 优化 静态 编译 (ahead-of-time, AOT) 处 理 
动态 编译 (just-in-time, JIT) 处 理 的 缺点 在 于 它 会 强制 用 户 等 待 操作 完成 。 但 是 激进 的 
AOT 处 理 则 会 导致 计算 资源 的 浪费 。 需 要 根据 应 用 和 设备 选择 精确 定量 的 AOT 处 理 。 


例如 ， 在 UITableView 中 演 染 一 组 记录 时 ， 在 载 和 列表 时 处 理 全 部 的 记录 并 不 是 明智 的 
选择 。 基 于 单元 格 的 高 度 ， 如 果 设 备 可 以 泻 染 X 条 记录 ， 那 么 3V 或 4V 则 是 一 个 理想 
的 数据 载 和 规模。 类 似 地 ， 如 果 用 户 快速 滚动 ， 则 不 应 立即 载 和 记录， 而 应 推迟 到 滚动 
速度 下 降 至 某 一 闽 值 。 精 确 的 阔 值 应 该 由 每 个 单元 格 的 处 理 时 间 和 单元 格 UI 的 复杂 性 
来 决定 〈 例 如 ， 单 元 格 中 包含 多 张 图 片 或 视频 ) 。 

。 分 析 电 量 消 耗 

测量 目标 用 户 的 所 有 设备 上 的 电量 消耗 (参见 3.7 节 )。 找 到 高 能 耗 的 区 域 并 想 办 法 降 
低能 


3.2 网络 


智能 的 网 络 访问 管理 可 以 让 应 用 响应 得 更 快 ， 并 有 助 于 延长 电池 寿命 。 在 无 法 访问 网 络 
时 ， 应 当 推迟 后 续 的 网 络 请 求 ， 直 到 网 络 连接 恢复 为 止 。 

此 外 ， 应 避免 在 没有 连接 WiFi 的 情况 下 进行 高 带宽 消耗 的 操作 ， 比 如 视频 流 。 众 所 周知 ， 
蜂窝 无 线 系统 (LTE、4G、3G 等 ) 对 电量 的 消耗 远大 于 WiFi 信号 。 根 源 在 于 LTE 设备 
基于 多 输入 、 多 输出 技术 ， 使 用 多 个 并 发 信号 以 维护 两 端的 LTE 链接 。 类 似 地 ， 所 有 的 蜂 
宽 数 据 连 接 都 会 定期 扫描 以 寻找 更 强 的 信号 。 

因此 ， 我 们 需要 : 

。 在 进行 任何 网 络 操作 之 前 ， 先 检查 合适 的 网 络 连 接 是 否 可 用 ， 

。 持续 监视 网 络 的 可 用 性 ， 并 在 连接 状态 发 生变 化 时 给 予 适当 的 反馈 。 

苹果 公司 提供 了 示例 代码 (http://apple.co/1Q3gRKL)， 以 检查 和 监听 网 络 状 态 的 变化 。 如 
果 你 的 项 目 使 用 了 CocoaPods， 那 么 请 使 用 Tony Million 的 Reachabilitypod (https:// 
github.com/tonymillion/Reachability ) 。 


例 3-1 展示 了 在 你 的 代码 中 添加 一 个 简单 的 方法 (isaPIServerAvailable), ， 并 在 真正 调用 
前 使 用 该 方法 。 


例 3-1 检查 网 络 状 态 


//Helper API 
-(BOOL)isAPIServerReachable { 
Reachability *r = [Reachability reachabilityWithHostname:@"api.yourdomain.com"];@ 
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return r.isReachable;@ 


j 
// 真 正 的 网 络 操作 


-(void)performNetworkOperation:(NSDictionary *)params 
completion:(void (^)(NSError *, id)) completion { © 


























if(!self.isAPIServerReachable) { 
[self enqueueRequest:params completion:completion]; @ 


NSError *err = [[NSError alloc] initWithDomain:Q"network" 
code:kErrorCodeNetworkUnreachable userInfo:nil]; 
completion(err, nil); © 
} else { 
[self doNetworkOperation:params completion:completion]; (9 


j 
} 


@ 检查 服务 器 域名 是 否 可 达 。 

@ 可 以 随意 地 对 isReachableViaWiFi 或 isReachableViaWWAN (3G, 4G, EDGE 等 ) 方法 
进行 优化 改造 。 

© completion 回调 方法 提供 了 (指定 操作 的 ) id 类 型 的 结果 或 NSError * 类 型 的 错误 。 

O 对 操作 进行 排队 。queue 方法 的 实现 并 没有 在 这 里 体现 出 来 。 

@ kErrorCodeNetworkUnreachable 是 在 应 用 中 定义 的 一 个 常量 。 

Q 如 果 网 络 可 用 ， 立 即 触发 请 求 。 

类 似 地 ， 为 了 实现 第 二 步 (监听 网 络 状 态 并 在 网 络 可 用 时 执行 队列 )， 你 可 以 使 用 例 3-2 中 

的 代码 。 


例 3-2 监控 网 络 并 执行 队列 


/ [HPNetworkOps .h 
Gproperty (nonatomic, readonly) BOOL isAPIServerReachable; 
































/ [HPNetworkOps .m 


(property (nonatomic, strong) Reachability *reachability; 
@property (nonatomic, strong) NSOperationQueue *networkOperationQueue; 


-(id)init { 
if(self = [super init]) { 
self.reachability = [Reachability 
reachabilityWithHostname:@"api. yourdomain.com" ]; 
self.reachability.reachableOnWWAN = NO; 


self.networkOperationQueue - [[NSOperationQueue alloc] init]; 
self.networkOperationQueue.maxConcurrentOperationCount - 1; 


[[NSNotificationCenter defaultCenter] addObserver:self 
selector :@selector (networkStatusChanged: ) 
name:kReachabilityChangedNotification object:nil]; 

} 


return self; 


} 


- (void )networkStatusChanged: (Reachability *)reachability { 
if(!reachability.isReachableViaWiFi) { 


self.networkOperationQueue.suspended - YES; 
) else { 
self.networkOperationQueue.suspended - NO; 
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j 


-(BOOL)isAPIServerReachable { 
return self.reachability.isReachableWiFi; 


-(void)performNetworkOperation:(NSDictionary *)params 
completion:(void (^)(NSError *, id)) completion { 
[self enqueueRequest:params completion:completion]; 


j 


-(void)enqueueRequest:(NSDictionary *)params 
completion:(void (^)(NSError *, id)) completion 


{ 
NSURLRequest *req = ...; 
AFHTTPRequestOperation *op = 
[[AFHTTPRequestOperation alloc] initWithRequest:req]; 
[op setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *op, id res){ 
completion(nil, res); 
) failure:^(AFHTTPRequestOperation *operation, NSError *error) { 
completion(error, nil); 
]]; 
[self.networkOperationQueue addOperation:op] 
} 


以 下 是 对 例 3-2 的 代码 分 析 。 


e 类 HPNetworkOps 有 一 个 isAPIServerReachable 属性 , 该 属性 可 以 用 于 检测 网 络 是 否 可 用 。 
取决 于 应 用 、 应 用 的 状态 或 者 具体 的 任务 ， 你 可 以 使 用 这 个 标记 来 判断 是 继续 处 理 
是 使 用 一 个 平视 显示 器 来 阻碍 应 用 的 交互 。 


。 这 个 类 具有 私有 属性 reachability 和 networkOperationQueue, 
€ reachability 属性 用 于 监听 状态 。 它 仅 用 于 跟踪 WiFi 网 络 的 变化 。 
4 networkOperationQueue 保留 了 队列 中 的 操作 。 该 队列 一 次 只 允许 执行 一 个 操作 。 
。 根据 网 络 的 可 用 情况 ， 通 知 的 接收 者 (networkstatusChanged) 挂 起 或 恢复 队列 。 
e 更 新 了 performNetworkOperation 的 实现 ， 从 而 总 是 将 网 络 操作 送 入 队列 中 。 
e enqueueRequest:completion: 方法 对 网 络 操作 进行 排队 。 
这 个 例子 中 的 AFHTTPRequestOperation 来 源 于 AFNetworkingpod (https://github.com/AFNet- 
working/AFNetworking)。 你 可 以 自由 选择 其 他 的 操作 或 创建 自己 的 操作 。 
在 这 段 代码 中 ， 我 们 用 NSOperationQueue 进行 了 演示 。 你 可 能 还 需要 一 个 更 加 复杂 的 队列 


来 实现 额外 的 控制 。 更 极端 一 点 的 话 ， 你 可 能 需要 将 暂停 的 网 络 操作 保存 起 来 ， 以 便 后 续 
网 络 恢复 可 用 时 再 与 服务 器 同步 更 新 。 


注意 ，NS0perationQueue 不 会 暂停 或 挂 起 任何 执行 中 的 操作 。 一 个 挂 起 的 队列 仅仅 意味 着 
后 续 操 作 在 其 恢复 之 前 不 会 被 执行 。 正 如 苹果 公司 的 开发 者 文档 所 描述 的 那样 。 
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操作 只 有 完成 后 才 会 从 队列 中 移 除 。 然 而 ， 为 了 完成 执行 ， 必 须 先 启动 操作 。 因 
为 挂 起 的 队列 不 会 启动 任何 新 的 操作 ， 所 以 它 也 不 会 移 除 任何 正在 排队 且 未 被 执 
行 的 操作 ( 包括 那些 已 经 取消 的 操作 )。 


使 用 基于 队列 的 网 络 请 求 以 避免 服务 器 被 多 个 同时 发 起 的 请 求 所 缀 炸 。 至 少 
使 用 两 个 队列 : 一 个 用 于 通常 不 是 很 关键 的 大 量 图 片 下 载 ， 另 一 个 用 于 关键 
数据 的 请 求 。 参 见 4.4 节 以 了 解 使 用 操作 的 详细 信息 。 
此 外 ， 作 为 一 位 有 情怀 的 工程 师 ， 你 还 需要 更 新 网 络 活动 指示 符 一 一 将 操 
作 加 入 队列 时 打开 它 ， 收 到 网 络 响应 后 将 其 关闭 。 这 里 需要 使 用 定义 在 
UIApplication 类 中 的 setNetworkActivityIndicatorVisible 方法 。 
































3.3 ”定位 管理 器 和 GPS 
了 解 定 位 服务 包括 GPS (或 GLONASS) 和 WiFi 硬件 这 一 点 很 重要 ， 同 时 要 知道 定位 服 
务 需要 大 量 的 电量 。 
使 用 GPS 计算 坐标 需要 确定 两 点 信息 。 
。 时 间 锁 
每 个 GPS 卫星 每 密 秒 广播 唯一 一 个 1023 位 随机 数 ， 因 而 数据 传播 速率 是 1.024Mbit/s, 
GPS 的 接收 芯片 必须 正确 地 与 卫星 的 时 间 锁 槽 对 齐 。 


GPS 接收 器 必须 计算 由 接收 器 与 卫星 的 相对 运动 导致 的 多 普 勒 偏 移 带 来 的 信号 误差 。 


通常 情况 下 ， 锁 定 一 颗 卫 星 至 少 需要 30 秒 。 必 须 锁 定 接收 范围 内 的 所 有 卫星 。 确 定 的 卫 
星 越 多 ， 取 得 的 定位 坐标 就 越 精确 。 


计算 坐标 会 不 断 地 使 用 CPU 和 GPS 的 硬件 资源 ， 因 此 它们 会 迅速 地 消耗 电池 电 
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GPS 精密 码 
卫星 的 精度 码 更 广为人知 的 名 字 是 P-code， 实 际 长 度 为 6.1871x102 位 ， 大 约 有 
720GB, ‘EVA 10.23Mbit/s 的 速度 传输 ， 每 周 重复 一 次 。 


更 有 趣 的 是 ， 这 个 P-code 只 是 一 个 主 P-code 的 子 集 。 主 P-code 的 长 度 接近 2.34 x 10" 
位 ， 大 约 有 26.7TB 











现在 你 已 经 了 解 了 GPS 锁定 过 程 的 复杂 本 质 ， 再 怎么 强调 要 对 其 小 心 使 用 都 不 为 过 ， 尤 其 
是 在 你 的 应 用 重度 依赖 地 图 的 情况 下 。 


接 下 来 我 们 将 讨论 可 以 最 小 化 电量 使 用 的 最 佳 实践 你 的 客户 一 定 会 因为 这 项 优化 而 格外 
满意 )。 


例 3-3 展示 了 初始 化 CLLocationManager 并 高 效 接收 地 理 位 置 更 新 的 典型 代码 。 


























fj 3-3 使 用 定位 管理 器 


//HPLocationViewController.h 
@interface HPLocationViewController : UIViewController <CLLocationManagerDelegate> 


(property (nonatomic, strong) CLLocationManager *manager; 
Qend 


//HPLocationViewController.m 
(implementation HPLocationViewController 


- (void) viewDidLoad 


{ 


self.manager = [[CLLocationManager alloc] init]; @ 
self.manager.delegate = self; 


} 
- (IBAction)enableLocationButtonTapped: (id) sender 
{ 
self .manager.distanceFilter = kCLDistanceFilterNone; @ 
self .manager.desiredAccuracy = kCLLocationAccuracyBest; © 
if(isI0s8()) (O 
[self.manager requestWhenInUseAuthorization]; G 
} 
[self.manager startUpdatingLocation]; 
} 


- (void) LocationManager :(CLLocationManager *)manager 
didUpdateLocations:(NSArray *)locations 


{ 
CLLocation *loc = [locations lastObject]; 
// 使 用 定位 信息 

} 

@end 


@ 这 里 没有 使 用 依赖 注入 ， 但 此 处 的 manager 被 视图 控制 器 所 持 有 、 管 理 和 配置 。 

四 初始 化 管理 器 ， 观 察 所 有 距离 的 变化 。 

Q9 按照 最 大 精度 初始 化 管理 器 。 

O 为 了 简化 代码 ， 这 里 省 略 了 isI0S8 辅助 方法 。 当 应 用 在 IOS 8 及 更 新 版 本 的 系统 上 运 



































行 时 ， 它 会 返回 true, 
6x 


是 iOS 8 特定 的 一 个 API， 用 于 在 应 用 活动 时 申请 使 用 定位 服务 。 





3.3.1 最 佳 的 初始 化 
正如 例 3-3 所 示 ， 在 调用 startUpdatingLocation 方法 时 ， 两 个 参数 起 着 非常 重要 的 作用 。 


























distanceFilter 
只 要 设备 的 移动 超过 了 最 小 距离 ， 距 离 过 滤器 就 会 导致 管理 器 对 委托 对 象 的 locationMa 
nager:didUpdateLocations: 事件 通知 发 生变 化 。 该 距离 使 用 公制 单位 CK), 





这 并 不 会 有 助 于 减少 GPS 接收 器 的 使 用 ， 但 会 影响 应 用 的 处 理 速度 ， 从 而 直接 减少 


CPU 的 使 用 。 


e desiredAccuracy 





精度 参数 的 使 用 直接 影响 了 使 用 天 线 的 个 数 ， 进 而 影响 了 对 电 凶 的 消耗 。 精 度 级 别 的 选 
取 取 决 于 应 用 的 具体 用 途 。 按 照 降序 排列 ， 精 度 由 以 下 常量 定义 。 


@ kCLLocationAccuracyBestForNavigation 


用 于 导航 的 最 佳 精度 级 别 。 
@ kCLLocationAccuracyBest 


设备 可 能 达到 的 最 佳 精度 级 别 。 





* kCLLocationAccuracyNearestTenMeters 





精度 接近 10 米 。 如 果 对 用 户 所 走 的 每 一 米 并 不 感 兴趣 ， 不 妨 使 用 这 个 值 ( 例 如 ， 可 


在 测量 大 块 距离 时 使 用 )。 





* kCLLocationAccuracyHundredMeters 


精度 接近 100 米 〈 在 计算 距离 时 ， 


* kCLLocationAccuracyKilometer 





这 个 值 需要 乘 以 100 米 )。 


精度 在 千 米 范 围 。 这 在 粗略 测量 两 个 距离 数 百 千 米 的 兴趣 点 时 非常 有 用 (例如 ， 如 
果 计 算 从 旧金山 的 家 到 阿 纳 海 姆 的 迪斯尼 乐园 的 距离 ， 这 个 精度 就 已 足够 )。 











* kCLLocationAccuracyThreeKilometers 





精度 在 3 千 米 范 围 。 在 距离 真 的 很 远 时 使 用 这 个 值 〈 如 果 计 算 位 于 英国 伦敦 的 家 到 
印度 泰 姬 陵 的 距离 ， 这 个 精度 也 就 足够 了 ) 。 


距离 过 滤器 只 是 软件 层面 的 过 滤器 ， 而 精度 级 别 会 影响 物理 天 线 的 使 用 。 

当 委 托 的 回调 方法 LocationManager:didUpdateLocations: 被 调用 时 ， 使 用 
距离 范围 更 广 的 过 渡 器 只 会 影响 间隔 。 男 一 方面 ， 更 高 的 精度 级 别 意 味 着 更 
多 的 活动 天 线 ， 这 会 消耗 更 多 的 能 量 。 














3.3.2 ”关闭 无 关 紧 要 的 特 





性 


判断 何 时 需要 跟踪 位 置 的 变化 。 在 需要 跟踪 时 调用 startUpdatingLocation 方法 ， 无 需 跟 


踪 时 调用 stopUpdatingLocation 方法 。 


假设 用 户 需要 用 一 个 消息 类 的 应 用 与 朋友 分 享 位 置 。 如 果 该 应 用 只 是 发 送 城市 的 名 称 ， 则 








只 需要 一 次 性 地 获取 位 置信 息 ， 然 后 就 可 以 通过 调用 stopUpdatingLocation 关闭 位 置 跟 
踪 。 在 一 定 的 时 间 间 隔 后 可 以 再 次 开启 定位 。 你 可 以 设置 固定 的 间隔 (如 60 秒 或 5 分 
钟 )， 也 可 以 动态 地 计算 时 间 间 隔 ( 例 如， 根据 之 前 获取 的 坐标 和 速度 ， 佑 算 穿 过 城市 的 


时 间 上 限 )。 





体 库 、 查 看 朋友 列表 或 调整 应 用 设置 时 





当 应 用 在 后 台 运 行 或 用 户 没有 与 别人 聊天 时 ， 也 应 该 关闭 位 置 跟踪 。 这 也 就 是 说 ， 浏 览 媒 


， 都 应 该 关闭 位 置 跟踪 。 


向 终端 用 户 提供 关闭 非 必要 功能 的 选项 是 一 个 更 好 的 解决 方案 。 例 如 ，Waze 应 用 提供 了 


一 个 选项 以 关闭 应 用 中 的 所 有 活动 (如 





图 3-1 所 示 )。 











5:26 PM 了 从 100% H+ 
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Waze is currently inactive. You are not 
using battery, contributing data, or 
earning points. 


Wake Up 











图 3-1: 关闭 了 位 置 跟 踪 的 Waze 应 用 


3.3.3 ”只 在 必要 时 使 用 

为 了 提高 电量 的 使 用 效率 ， mbi. 能 地 保持 无 线 网 络 关 闭 。 
fet, iOS 会 利用 这 个 机 会 向 后 台 uda 话 ， 以 便 一 些 低 优 先 级 的 事件 能 够 被 处 
里 ， 如 推送 通知 、 ee 

关键 在 于 每 当 应 用 建立 网 络 连接 上 时， 网 络 硬件 都 会 在 连接 完成 后 多 维持 儿 秒 的 活动 时 间 。 
每 次 集中 的 网 络 通 信和 都 会 消耗 大 量 的 电量 。 

要 想 减轻 这 个 问题 带 来 的 危害 ， 你 的 软件 需要 有 所 保留 地 使 用 网 络 。 应 该 定期 集中 短暂 地 
使 用 网 络 ， 而 不 是 持续 地 保持 着 活动 的 数据 流 。 只 有 这 样 ， 网 络 硬件 才 有 机 会 被 关闭 。 
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3.3.4 后 台 定 位 服务 


CLLocationManager 提供 了 一 个 替代 的 方法 来 监听 位 置 的 更 新 。startMonitoringSigni- 
ficantLocationChanges 可 以 帮助 你 在 更 远 的 距离 跟踪 运动 。 精 确 的 值 由 内 部 决定 ， 且 与 
distanceFilter 无 关 。 


使 用 这 一 模式 可 以 在 应 用 进入 后 台 后 继续 跟踪 运动 。 (除非 应 用 是 导航 类 应 用 ， 且 你 想 在 锁 
屏 期 间 也 获得 很 好 的 细节 。) 典型 的 做 法 是 在 应 用 进入 后 台 时 执行 startMonitoringSigni- 
ficantLocationChanges 方法 ， 而 当 应 用 回 到 前 台 时 执行 startUpdatingLocation。 你 可 以 
在 自己 的 应 用 中 使 用 例 3-4 中 的 示例 代码 。 


例 3-4 监听 与 重大 变化 监听 


// 应 用 委托 
- (void)applicationDidEnterBackground: (UIApplication *)application 














[self.locationManager stopUpdatingLocation]; 
[self.locationManager startMonitoringSignificantLocationChanges]; 
- (void)willEnterForeground:(UIApplication *)application 


[self.locationManager stopMonitoringSignificantLocationChanges]; 
[self.locationManager startUpdatingLocation]; 





É iOS 8 系统 中 的 后 台 使 用 定位 
当 应 用 位 于 iOS 8 中 的 后 台 时 ， 你 需要 显 式 地 申请 权限 来 使 用 定位 管理 器 。 为 了 获取 
用 户 的 授权 ， 必 须 完 成 以 下 步骤 。 
(1) 更 新 应 用 的 Info.plist 文件 ， 添 加 String 类 型 的 NSLocationAlwaysUsageDescription 
条 目 。 相 应 的 值 是 应 用 进入 后 台 或 前 台 时 向 用 户 申请 权限 时 显示 的 消息 。 当 应 用 只 
在 前 台 时 ， 使 用 NSLocationNhenInUseUsageDescription 获取 使 用 位 置 的 权限 。 


(2) 调用 requestAlwaysAuthorization 方法 申请 前 台 和 后 台 权 限 (requestWhenInUseAuthorization 
方法 仅 适 用 于 前 台 使 用 )。 














5 iOS 7 不 同 ， Muri pon Jj E iOS 8 中 不 再 申请 用 户 权 
限 以 便 使 用 定位 数据 。 你 必须 使 用 requestWhenInUseAuthorization 或 


requestAlwaysAuthorization, 





3.3.5 ”NSTimer、NSThread 和 定位 服务 


当 应 用 位 于 后 台 时 ， 任 何 定 时 器 或 线程 都 会 挂 起 。 但 如 果 你 在 应 用 位 于 后 台 状 态 时 申请 了 
定位 ， 那 么 应 用 会 在 每 次 收 到 更 新 后 被 短暂 唤醒 。 在 此 期 间 ， 线 程 和 计时 器 都 会 被 唤醒 。 


























可 怕 之 处 在 于 ， 如 果 你 在 这 段 时 间 做 了 任何 网 络 操作 ， 则 会 启动 所 有 相关 的 天 线 (如 WiFi 
和 LTE/4G/3G ) 。 


想 要 控制 这 种 状况 往往 非常 坏 手 。 最 佳 的 选择 是 使 用 NSURLSession 类 。 我 们 会 在 7.1.5 市 
讨论 “网 络 AP", 


3.8.6 在 应 用 关闭 后 重启 


ci LEE 要 更 多 资源 时 ， 后 台 的 应 用 可 能 会 被 关闭 。 在 这 种 情况 下 ， 
一 旦 发 生 位 置 变化 ， 应 用 会 被 重启 ， 因 而 需要 重新 初始 化 监听 过 程 。 若 出 现 这 种 情况 ，a 
plication:did es : 方法 会 收 到 键 值 为 UIApplicationLaunchOption 
sLocationKey 的 条 目 。 相 关 代码 见 例 3-5。 


例 3-5 在 应 用 关闭 后 重新 初始 化 监听 
-(void)application:(UIApplication *)app 
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions 
{ 
if(launchOptions[UIApplicationLaunchOptionsLocationKey]) ( @ 
[self.manager startMonitoringSignificantLocationChanges]; @ 





















































j 
j 


O 办 缺乏 资源 而 关闭 应 用 后 ， 检 测 应 用 是 否 因为 位 置 变化 而 被 重启 了 。 
O 如 果 是 这 样 的 话 ， 开 始 监听 位 置 的 变化 。 如 果 不 是 ， 则 在 后 续 适 合 的 时 机 开始 监听 。 


3.4 屏幕 


屏幕 非常 耗 电 。 屏 幕 越 大 就 越 费 电 。 当 然 ， 如 果 你 的 应 用 在 前 台 运 行 且 与 用 户 进行 交互 ， 
则 势必 会 使 用 屏幕 并 消耗 电量 。 


然而 ， 仍 然 有 一 些 方案 可 以 优化 屏幕 的 使 用 。 


3.4.1 动画 
明智 地 使 用 动画 是 一 个 被 遗弃 的 概念 。 尽 管 如 此 ， 我 们 讨论 这 个 问题 是 为 了 表述 的 完整 性 。 


你 可 以 遵守 一 个 简单 的 规则 : 当 应 用 在 前 台 时 使 用 动画 ， 一 旦 应 用 进入 后 台 则 立即 暂停 动 
画 。 通 常 来 说 ， 你 可 以 通过 监听 UIApplicationWillResignActiveNotification 或 UIApplic 
ationDidEnterBackgroundNotification 的 通知 事件 来 暂停 或 停止 动画 ， 也 可 以 通过 监听 UI 
ApplicationDidBecomeActiveNotification 的 通知 事件 来 恢复 动画 。 


3.4.2 ”视频 播放 


在 视频 播放 期 间 ， 最 好 强制 保持 屏幕 常 亮 。 可 以 使 用 UIApplication 对 象 的 idleTimerDisabled 
属性 来 实现 这 个 目的 。 一 旦 设置 为 YYS， 它 会 阻止 屏幕 休眠 ， 从 而 实现 常 亮 。 


与 动画 类 似 ， 你 可 以 通过 响应 应 用 的 通知 来 释放 和 获取 锁 。 






































3.4.3 多 屏幕 
使 用 屏幕 比 休眠 锁 或 暂停 /恢复 动画 要 复杂 很 多 。 
如 果 设 备 连接 了 外 部 显示 设备 使 用 AirPlay 或 HDMI 22245), IBAA RAE ANE? 大 部 
分 应 用 通常 表现 为 操作 系统 的 默认 行为 ， 即 以 镜像 方式 向 外 部 显示 设备 投影 。 
但 实际 上 还 可 以 做 得 更 多 。 如 果 正 在 播放 电影 或 运行 动画 ， 你 可 以 将 它们 从 设备 的 屏幕 挪 
到 外 部 屏幕 ， 而 上 只 在 设备 的 屏幕 上 保留 最 基本 的 控制 。 这 样 可 以 减少 设备 上 的 屏幕 更 新 ， 
进而 延长 电池 寿命 。 侠 果 公 司 的 开发 者 网 站 提供 了 使 用 外 部 屏幕 (http:/apple.co/ljauUnu) 
的 简单 示例 。 使 用 数据 线 将 iPhone 或 iPad 连接 到 汽车 的 屏幕 或 使 用 AirPlay 将 它们 连接 到 
AppleTV， 并 非 是 罕见 的 事情 。 
处 理 这 一 场景 的 典型 代码 会 涉及 以 下 步骤 。 
(1) 在 启动 期 间 检测 屏幕 的 数量 。 

如 果 屏 幕 数量 大 于 1， 则 进行 切换 。 
(2) 监听 屏幕 在 连接 和 断 开 时 的 通知 。 

如 果 有 新 的 屏幕 加 入 ， 则 进行 切换 。 

如 果 所 有 的 外 部 屏幕 都 被 移 除 ， 则 恢复 到 默认 显示 。 
例 3-6 展示 了 如 何 通 过 多 个 屏幕 获取 效益 。 


Gil 3-6 使 用 多 个 屏幕 
//HPMultiScreenViewController.m 
@interface HPMultiScreenViewController () 



























































@property (nonatomic, strong) UIWindow *secondWindow; 
@end 

@implementation HPMultiScreenViewController 

- (void) viewDidLoad 


[super viewDidLoad]; 
[self registerNotifications]; 


} 
- (void)viewDidAppear : (BOOL) animated 
{ 
[super viewDidAppear : animated]; 
[self updateScreens]; 
} 


- (void) viewDidDisappear : (BOOL) animated 


[super viewDidDisappear : animated]; 
[self disconnectFromScreen]; 
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-(void)disconnectFromScreen 























{ 
if(self.secondWindow != nil) { 
// 断 开 链接 并 准备 释放 内 存 
self.secondWindow.rootViewController = nil; 
self.secondWindow.hidden - YES; 
self.secondWindow = nil; 
} 
} 
- (void) updateScreens 
{ 
NSArray *screens = [UIScreen screens]; 
if(screens.count > 1) { 
UIScreen *secondScreen = (UIScreen *)[screens objectAtIndex:1]; 
CGRect rect = secondScreen.bounds; 
if(self.secondWindow == nil) ( 
self.secondWindow = [[UIWindow alloc] initWithFrame:rect]; 
self.secondWindow.screen = secondScreen; 
HPScreen2ViewController *svc - [[HPScreen2ViewController alloc] init]; 
// 设 置 svc 的 其 他 属性 以 完整 地 对 它 进 行 初始 化 
svc.parent = self; 
self.secondWindow.rootViewController = svc; 
} 
self.secondWindow.hidden = NO; 
} else { 
[self disconnectFromScreen]; 
} 
} 
-(void)dealloc 
{ 
[self unregisterNotifications ]; 
} 
-(void)screensChanged:(NSNotification *)notification 
{ 
[self updateScreens]; 
} 


-(void)registerNotifications 


NSNotificationCenter *nc = [NSNotificationCenter defaultCenter]; 
[nc addObserver:self 


j 


selector :@selector(screensChanged: ) 
name:UIScreenDidConnectNotification object:nil]; 
[nc addObserver:self 
selector :@selector(screensChanged: ) 
name:UIScreenDidDisconnectNotification object:nil]; 


-(void)unregisterNotifications 








[[NSNotificationCenter defaultCenter] removeObserver:self]; 


} 


@end 

















在 例 3-6 rh, HPMultiscreenVtewController 是 包含 视频 播放 或 动画 UI 的 视图 控制 器 。 
在 此 示例 中 ， 我 们 使 用 了 另外 一 个 辅助 视图 控制 器 一 一 HPScreen2ViewController。 它 会 和 














父 视 图 控制 器 通信 ， 并 根据 用 户 的 交互 发 送 适 当 的 消息 。 以 下 是 对 每 个 方法 的 详细 介绍 。 











viewDidLoad 

因为 这 个 方法 在 视图 控制 器 的 生命 周期 中 会 且 仅 会 被 调用 一 次 ， 所 以 它 就 成 为 了 注册 
UIScreenDidConnectNotification (屏幕 连接 ) 通知 和 UIScreenDidDisconnectNotifica- 
tion (屏幕 断 开 ) 通知 的 最 佳 位 置 。 

每 当 有 新 的 屏幕 加 入 或 有 屏幕 移 除 时 ， 我 们 都 会 调用 screensChanged: 方法 ， 在 这 里 更 
新 UI。 

viewDidAppear: 

由 于 视图 可 以 显示 或 消失 多 次 ,或 者 更 具体 来 说 ， 用 户 可 以 进入 或 离开 视图 多 次 ， 我 们 
可 以 用 这 个 方法 更 新 屏幕 。 

用 户 首 次 进入 视图 控制 器 时 ，UI 会 进行 调整 ， 查 看 可 用 的 屏幕 数量 。 类 似 地 ， 如 果 用 
户 离开 这 个 视图 控制 器 ， 转 而 进入 其 他 视图 控制 器 ， 然 后 再 返回 ， 此 时 屏幕 数量 可 能 
经 发 生变 化 。 因 此 ， 它 们 可 能 需要 进行 调整 。 

viewDidDisappear: 

当 用 户 离开 这 个 视图 控制 器 时 ， 你 可 能 也 想 在 另 一 个 屏幕 上 更 新 UI。 使 用 这 个 方法 就 
能 实现 这 一 操作 。 
在 上 述 示例 中 ， 我 们 从 屏幕 中 移 除 了 secondWindow (通过 disconnectFromScreen 方法 )。 
在 一 个 更 加 复杂 的 应 用 中 ， 你 可 能 需要 支持 在 外 部 屏幕 上 播放 视频 的 同时 允许 用 户 进行 
复杂 的 操作 ， 例 如 ， 在 设备 屏幕 上 重新 排列 播放 列表 或 搜索 媒体 。 


图 3-2 展示 了 使 用 类 似 应 用 时 的 模拟 UI。 
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3-2; BA UI 





disconnectFromScreen 


调用 这 个 方法 将 secondwindow 从 屏幕 上 移 除 。 

updateScreens 

这 是 真正 的 魔法 产生 的 地 方 ， 尽 管 示例 出 于 演示 目的 极力 保持 简洁 。 

在 这 个 方法 中 ， 我 们 检查 了 屏幕 的 数量 ， 如 果 数 量 大 于 1， 我 们 就 将 新 的 窗口 与 第 二 屏 
连接 。 在 实际 使 用 时 ， 你 可 以 扫描 遍历 所 有 的 屏幕 ， 并 决定 对 哪 一 个 屏幕 进行 操作 一 一 
最 简单 的 操作 就 是 将 UI 复制 到 所 有 屏幕 上 。 

如 果 屏 幕 数 是 1， 则 调用 disconnectFromScreen 方法 。 

dealloc 

当 视 图 控制 器 即将 被 销毁 时 调用 一 次 ， 我 们 用 这 个 方法 解除 对 屏幕 通知 的 广 册 。 


screensChanged: 


当 应 用 获取 屏幕 断 开 的 通知 时 ， 该 方法 调用 updateScreens, 


registerNotifications 
文 个 方法 添加 了 UIScreenDidConnectNotification 和 UIScreenDidDisconnectNotifica- 


tion 通知 的 观察 者 。 





























unregisterNotifications 

这 个 方法 移 除了 观察 者 。 

在 实际 的 应 用 中 ，HPScreen2ViewController 将 由 与 用 户 交 互 以 实现 控制 的 UI 组 成 ， 如 电 
影 播放 器 。 你 可 能 也 会 在 不 同 的 屏幕 间 切 换 控 制 器 。 例 3-7 展示 了 如 何 实现 切换 。 


例 3-7 在 屏幕 之 间 交 换 UI 


-(void)swapScreens(UIWindow *)currentWindow newWindow: (UIWindow *)newWindow 








NSArray *screens = [UIScreen screens]; 


UIScreen *deviceScreen = [screens objectAdIndex:0]; 
UIScreen *extScreen = [screens objectAdIndex:1]; 


// 你 可 以 视 情 况 合 理 地 设置 边界 
currentWindow.screen = extScreen; 
newWindow.screen = deviceScreen; 





在 一 个 屏幕 上 显示 却 在 另 一 个 屏幕 上 控制 似乎 显得 有 些 策 拙 ， 但 这 样 可 以 实 
现 不 中 断 的 显示 。 此 外 ， 如 果 控 制 是 标准 按钮 (如 播放 、 和 暂停 、 恢 复 、 停 止 
等 )， 实 际 体验 并 不 会 变 差 。 

当然 ， 不 要 在 交互 游戏 中 使 用 这 种 方法 ， 因 为 交互 游戏 需要 在 屏幕 上 进行 操 
作 。 如 果 你 这 人 么 做 了 ， 用 户 会 十 分 不 满 ， 因 为 他 们 无 法 在 一 个 空白 的 屏幕 上 
动态 地 操控 游戏 。 

















3.5 “其 他 硬件 


当 应 用 进入 后 台 时 ， 应 该 释放 对 这 些 硬 件 的 锁定 : 





edem 
。 相机 
。 扬声器 ， 除 非 应 用 是 音乐 类 的 
。 麦克 风 





我 们 并 不 会 在 这 里 讨论 这 些 硬 件 的 特性 ， 但 是 基本 规则 是 一 致 的 一 一 只 有 当 应 用 处 于 前 台 
时 才 与 这 些 硬件 进行 交互 ， 应 用 处 于 后 台 时 应 停止 交互 。 

扬声器 和 无 线 蓝牙 可 能 是 例外 。 如 果 你 正在 开发 音乐 、 收 音 机 或 其 他 的 音频 类 应 用 ， 则 需 
要 在 应 用 进入 后 台 后 继续 使 用 扬声器 。 不 要 让 屏幕 仅仅 为 音频 播放 的 目的 而 保持 常 亮 。 类 
似 地 ， 若 应 用 还 有 未 完成 的 数据 传输 ， 则 需要 在 应 用 进入 后 台 后 持续 使 用 无 线 监 牙 ， 例 
如 ， 与 其 他 设备 传输 文件 。 


3.6 ”电池 电量 与 代码 感知 


一 个 智能 的 应 用 会 考虑 到 电池 的 电量 和 自身 的 状态 ， 从 而 决定 是 否 要 真正 执行 资源 密集 消 
耗 型 的 操作 。 另 外 一 个 有 价值 的 点 是 对 充电 的 判断 ， 确 定 设备 是 否 处 于 充电 状态 。 


使 用 UIDevice 实例 可 以 获取 batteryLevel 和 batterystate (充电 状态 )。 你 可 以 将 例 3-8 
中 的 代码 直接 用 于 自己 的 应 用 。ceedwithMinLevel: 方法 传 入 执行 特定 操作 需要 的 最 低 电 量 
级 别 。 该 级 别 是 浮 点 数 ， 范 围 在 0~100 (100 表示 电池 完全 充满 ) 。 


例 3-8 使 用 电量 级 别 和 充电 状态 进行 条 件 处 理 
- (BOOL) shouldProceedWithMinLevel:(NSUInteger)minLevel 
{ 


UIDevice *device = [UIDevice currentDevice]; 
device. batteryMonitoringEnabled = YES; 













































































UIDeviceBatteryState state = device.batteryState; 

if(state == UIDeviceBatteryStateCharging | | 
state == UIDeviceBatteryStateFull) { @ 
return YES; 


} 


NSUInteger batteryLevel = (NSUInteger) (device.batteryLevel * 100); @ 
if(batteryLevel >= minLevel) { 

return YES; 
} 


return NO; 


} 


@ 在 充电 或 电池 已 经 充满 的 情况 下 ， 任 何 操作 都 可 以 执行 。 
Q UIDevice 返回 的 batteryLevel 的 范围 在 0.00-1.00, 


类 似 地 ， 你 还 可 以 得 到 应 用 对 CPU 的 利用 率 。 在 应 用 运行 时 这 可 能 并 不 是 一 个 特别 重要 





























的 信息 ， 但 是 出 于 完整 性 的 目的 ， 我 们 在 例 3-9 中 给 出 了 这 部 分 代码 。 
例 3-9 应 用 对 CPU 的 使 用 率 


-(float)appCPUUsage { 
kern return t kr; 
task info data t info; 
mach msg type number t infoCount - TASK INFO MAX; 





kr = task info(mach task self(), TASK BASIC INFO, 
(task info t)info, &infoCount); 
if (kr != KERN SUCCESS) { 


return -1; 
} 
thread_array_t thread_list; 
mach_msg_type_number_t thread_count; 
thread_info_data_t thinfo; 
mach_msg_type_number_t thread_info_count; 
thread_basic_info_t basic_info_th; 


kr = task_threads(mach_task_self(), &thread_list, &thread_count); 
if (kr !- KERN SUCCESS) { 
return -1; 


j 


float tot cpu - 0; 
int j; 


for (j = 0; j < thread count; j++) { 
thread info count - THREAD INFO MAX; 
kr = thread info(thread list[j], THREAD BASIC INFO, 
(thread info t)thinfo, &thread info count); 
if (kr !- KERN SUCCESS) { 
return -1; 


} 
basic_info_th = (thread_basic_info_t)thinfo; 
if (!(basic_info_th->flags & TH FLAGS IDLE)) { 


tot cpu += basic info th-»cpu usage / 
(float)TH USAGE SCALE * 100.0; 


} 


vm_deallocate(mach_task_self(), (vm_offset_t)thread_list, 
thread_count * sizeof(thread_t)); 
return tot_cpu; 




















当 剩余 电量 较 低 时 提示 用 户 ， 并 请 求 用 户 授 权 执 行 电源 密集 型 的 操作 一 一 当 
然 ， 只 在 用 户 同意 的 前 提 下 执行 。 
总 是 用 一 个 指示 符 显示 长 时 间 任 务 的 进度 ， 包 括 设备 上 即将 完成 的 计算 或 者 
只 是 下 载 一 些 内 容 。 向 用 户 提 供 完 成 进度 的 估算 ， 以 帮助 他 们 决定 是 否 需 
为 设备 充电 。 





























3.7 ”分析 电量 使 用 


在 开发 期 间 使 用 Xcode Instruments 跟踪 CPU 的 使 用 。 我 们 将 在 第 11 章 对 该 工具 进行 深入 
探讨 。 现 在 我 们 对 其 中 的 活动 监视 器 (参见 11.2.2 节 ) 模板 更 感 兴趣 。 它 很 好 地 提供 了 测 
量 电量 消耗 的 能 力 ， 因 为 CPU 消耗 了 最 多 的 电量 。 

要 想得到 准确 量化 的 应 用 的 电量 消耗 ， 你 可 以 使 用 Monsoon Solutions 的 电源 监控 器 
(Power Monitor, https://www.msoon.com/LabEquipment/PowerMonitor), 。 使 用 这 个 工具 的 步 
WAT. 

(1) 拆 开 iOS 设备 的 外 这， 找到 电池 后 面 的 电源 针脚 。 

(2) 连接 电源 监控 器 的 设备 针脚 。 

(3) 运行 应 用 。 

(4) 测量 电量 消耗 。 

图 3-3 展示 了 与 iPhone 的 电池 针脚 连接 的 电源 监控 器 工具 。 


























图 3-3:; 电源 监控 器 与 iPhone 5S 进行 连接 (图 片 摘自 Bottle of Code) 


电源 监控 器 软件 可 以 跟踪 电源 随时 间 流 逝 的 使 用 情况 。 数 据 以 图 表 的 形式 展示 出 来 了 ， 如 
图 3-4 所 示 。 


























= 
«Q Power Tool --TempSaved-- 


Measured Power Data Vout ENABLED 
2.99 V 


Inst Current 
1.38 mA [o | 








Power Avg 
Power Min 
Power Max 











Time 11811 s 
Samples 590550 
PUE OTHER Consumed Energy — 4629 uAh 




















Average Power — 422 mW 
Average Curent — 141 mA 
Average Voltage — 288 V 
Expected Battery Life 18948 hrs 
225 mh Battery 

© Main 
CURSORS 

Value 








Noe w 





114 CAPTURE TRIGGERS 
Time(s) START: Manual 




















图 3-4: 电源 监控 器 软件 


3.8 最 佳 实 践 


以 下 的 最 佳 实践 可 以 确保 对 电量 的 谨慎 使 用 。 遵 人 循 以 下 要 点 ， 应 用 可 以 实现 对 电量 的 高 间 
使 用 。 


。 最 小 化 硬件 使 用 。 换 句 话 说 , 尽 可 能 晚 地 与 硬件 打交道 , 并 且 一 旦 完成 任务 立即 结束 使 用 。 
。 在 进 (EMI. mie, 


。 在 电量 低 时 ， 提 示 用 户 是 否 确定 要 执行 任务 ， 并 在 用 户 同 意 后 再 执行 。 
|a occ ML oq 


例 3-10 Was Bl ACR Ran Y VES ea HI. REALI AE ILE 3-5, 
例 3-10 如果 电量 低 ， 在 执行 密集 型 操作 之 前 提示 用 户 


-(IBAction)onIntensiveOperationButtonClick:(id)sender { 





ig 























NSUserDefaults *defaults - [NSUserDefaults standardUserDefaults]; 
BOOL prompt = [defaults boolForKey:Q"promptForBattery"]; 
int minLevel = [defaults integerForKey:@"minBatteryLevel" ]; 


BOOL canAutoProceed = [self shouldProceedWithMinLevel:minLevel]; 
if(canAutoProceed) { 


[self executeIntensiveOperation]; 
} else { 
if(prompt) f 
UTAlertView *view = [[UIAlertView alloc] initWithTitle:Q"Proceed" 
message:@"Battery level below minimum required. Proceed?" 


delegate:self cancelButtonTitle:@"No" 
otherButtonTitles:@"Yes", nil]; 


[view show]; 
) else ( 


[self queueIntensiveOperation]; 








- (void)alertView:(UIAlertView *)alertView 
clickedButtonAtIndex:(NSInteger)buttonIndex { 


if(buttonIndex == 0) { 
[self queueIntensiveOperation]; 
} else { 
[self executeIntensiveOperation]; 


j 
} 


可 以 用 以 下 方式 解读 例 3-10 中 的 代码 。 


。 点 击 按钮 (或 任何 其 他 逻辑 ) 
该 方法 就 会 触发 密 集 型 的 操作 。 














就 会 执行 onIntensive0perationButtonCLick: 方法 。 假 设 


。 设置 由 两 个 条 目 组 成 : promptForBattery (应 用 设置 中 的 拨 动 开关 ， 表 明 是 否 要 在 低 电 
量 时 给 予 提 示 ) 和 miniBatteryLevel (区 间 为 0-100 的 一 个 请 块 ， 表 明了 最 低 电 量 



































在 此 示例 中 ， 用 户 可 以 自行 调整 )。 应 用 设置 的 界面 如 图 3-5 所 示 。 
0000 AT&T F 9:32 PM 9 75% ED + 
€ Settings HPerf Apps 
BATTERY 


Prompt for Battery 


LOCAL NOTIFICATIONS 


PONY DEBUGGER 


Enabled 


Network Debugging 


Core Data 





URL ws://10.73.218.219:9000/device 


Last 2015-06-08 05:22:01 +0000 
LOGGING 

Level Verbose 
VC DELAY 


b. / 
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\ 








图 3-5: 电量 水 平 的 阅 值 和 提示 选项 在 应 用 中 的 设置 
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在 实际 的 应 用 中 ， 应 用 的 开发 人 员 通 常 根 据 操作 的 复杂 性 和 密集 性 对 国 值 进行 预 设 。 不 

同 的 密集 型 操作 可 能 会 有 不 同 的 最 低 电量 需求 。 

。 在 实际 执行 密集 操作 之 前 ， 检 查 当 前 电量 是 否 足够 ， 或 者 手机 是 否 正 在 充电 。 这 就 是 我 
判断 是 否 可 以 进行 后 续 处 理 的 逻辑 ， 如 例 3-8 所 示 。 你 可 以 有 自己 的 定制 一 一 最 低 电量 
和 充电 状态 。 

。 如 果 条 件 有 具备 ， 则 立即 执行 (这 里 需要 在 类 中 调用 类 似 executeIntensiveOperation 的 
方法 )。 

。 否则 的 话 ， 如 果 用 户 设置 了 promptForBattery， 他 会 被 提示 。 

若 用 户 没 有 选择 被 提示 ， 我 们 将 密集 操作 放 入 队列 ， 以 便 后 续 执 行 ( 这 里 在 类 中 调用 类 
似 queueIntensiveOperation 的 方法 )。 

。 提示 之 后 ， 如 果 用 户 选择 Ok， 那 么 我 们 调用 executeIntensiveOperation， 否 则 调用 

queueIntensiveOperation, 


3.9 小结 

因为 用 户 总 会 随身 携带 移动 设备 ， 所 以 编写 省 电 的 代码 显得 格外 重要 ， 毕 竟 移 动 设备 的 充 
电 接 口 并 非 随处 可 见 ， 而 且 也 不 是 所 有 的 用 户 都 会 随身 携带 移动 电源 。 

在 无 法 降低 任务 复杂 性 (例如 ， 处 理 图 片 或 绘制 图 表 ) 时 ， 提 供 一 个 对 电池 电量 保持 敏感 
的 方案 并 在 适当 的 时 机 提示 用 户 ,会 让 用 户 感觉 良好 ， 并 因此 欣赏 你 的 应 用 。 

在 下 一 章 中 ， 我 们 将 讨论 并 行 执 行 多 个 任务 的 方案 和 最 佳 实践 。 本 章 中 优化 内 存 、 降 低能 
耗 的 内 容 将 为 我 们 进一步 讨论 的 话题 提供 素材 。 


































































































第 4 章 


HA Snitz 





iOS 设备 有 两 或 三 个 CPU 核心 ( 见 表 3-1)。 这 意味 着 ， 即 使 应 用 的 主线 程 (UI 线程) iE 
忙于 更 新 屏幕 ， 应 用 仍然 可 以 在 后 台 进 行 更 多 计算 ， 而 无 需 任 何 上 下 文 的 切换 。 
在 本 章 中 ， 我 们 将 探索 诸多 充分 利用 现 有 CPU 核心 的 方案 ， 并 学 习 如 何 通过 并 发 编程 优 
化 性 能 。 我 们 将 讨论 以 下 主题 : 

。 创建 和 管理 线程 

。 多 线程 优化 技术 (Grand Central Dispatch, GCD) 概述 

。 操作 和 队列 

我 们 将 讨论 编写 线程 安全 的 高 性 能 代码 的 技术 和 最 佳 实践 。 


4.1 线程 

线程 是 运行 时 执行 的 一 组 指令 序列 。 

每 个 进程 至 少 应 包含 一 个 线程 。 在 10S 中 ， 进 程 启动 时 的 主要 线程 通常 被 称 作 主 线程 。 所 
有 的 Ul 元 素 都 需要 在 主线 程 中 创建 和 管理 。 与 用 户 交 互相 关 的 所 有 中 断 最 终 都 会 分 发 到 
UI 线程 ， 处 理 代码 会 在 这 些 地 方 执行 一 一 IBAction 方法 的 代码 都 会 在 主线 程 中 执行 。 
Cocoa 编程 不 允许 其 他 线程 更 新 UT 元 素 。 这 意味 着 ， 无 论 何 时 应 用 在 后 台 线 程 执行 了 耗 时 
操作 ， 比 如 网 络 或 其 他 处 理 ， 代 码 都 必须 将 上 下 文 切换 到 主线 程 再 更 新 UI 一 一 例如 ， 进 度 
条 指示 任务 进度 或 标签 展示 处 理 结果 。 
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4.2 ”线程 开销 
uu NH E 但 每 个 线程 都 有 一 定 的 开销 ， TATUR BERI RE RUE 
E。 线 程 不 仅仅 有 创建 时 的 时 间 开 销 ， 还 会 消耗 内 核 的 内 存 ， 即 应 用 的 内 存 空 间 。 


4.2.1 内 核 数据 结构 


每 个 线程 大 约 消 耗 IKB 的 内 核 内 存 空间 。 a 
性 。 这 块 内 存 是 联动 内 存 (wired memory) ， 无 法 被 分 


4.2.2 zE] 

主线 程 的 栈 空间 大 小 为 LIM， 而 且 无 法 修改 。 所 有 的 二 级 线程 默认 分 配 512KB 的 栈 空 间 。 
注意 ， 完 整 的 栈 并 不 会 立即 被 创建 出 来 。 实 际 的 栈 空 间 大 小 会 随 着 使 用 而 增长 。 因 此 ， 即 
使 主线 程 有 IMB 的 栈 空 间 ， 某 个 时 间 点 的 实际 栈 空间 很 可 能 要 小 很 多 。 


在 线程 启动 前 ， 栈 空间 的 大 小 可 以 被 改变 。 栈 空间 的 最 小 值 是 16KB ， 而 且 其 数值 必须 是 
4KB 的 倍数 。 例 4-1 中 的 示例 代码 展示 了 如 何在 启动 线程 前 配置 栈 大 小 。 


il 4-1 修改 栈 空间 
+(NSThread *)createThreadWithTarget:(id)target selector:(SEL)selector 
object:(id)argument stackSize:(NSUInteger)size { 
























































if( (size % 4096) != 0) { 
return nil; 


NSThread *t - [[NSThread alloc] initWithTarget:target 
selector:selector object:argument]; 
t.stackSize - size; 


return t; 


j 


4.2.8 创建 耗 时 

我 们 在 iPhone 6 Plus iOS 8.4 上 进行 了 一 项 快速 测试 ， 展示 了 线程 创建 的 耗 时 (不 包含 启动 
时 间 )， 其 区 间 范 围 在 4000~5000 微 秒 ， 即 4~5 毫秒 。 

创建 线程 后 启动 线程 的 耗 时 区 间 为 5~100 上 毫秒， 平均 大 约 在 29 毫秒 。 这 是 很 大 的 时 间 开 
销 ， 若 在 应 用 启动 时 开启 多 个 线程 ， 则 尤为 明显 。 

线程 的 启动 时 间 之 所 以 如 此 之 长 ， 是 因为 多 次 的 上 下 文 切换 所 带 来 的 开销 。 

出 于 简洁 的 目的 ， 我 们 省 略 了 计算 的 代码 。 要 想 了 解 细 节 ， 你 可 以 参考 GitHub (https:// 
github.com/gvaish/hpios/blob/master/src/ ViewControllers/HPChapter05 ViewController.m) 中 的 
computeThreadCreationTime 方法 。 图 4-1 展示 了 这 段 代 码 的 输出 。 

































































iE 1: iOS Developer Library, “Thread Costs" (https://sites.google.com/site/appleiotemp//1EukJhy). 
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4.3 GCD 


GCD API (https://developer.apple. Soni beat yiinae idee ume ntatl onl Pertormance Reterenee! 


GCD_libdispatch_Ref/index.html) 由 核心 语言 特性 、 


增强 所 组 成 。 
我 们 不 打算 介绍 使 用 GCD 的 基 








E 础 知识 ， 因 为 那 不 是 本 二 


区 的 目的 。 你 应 


应 该 已 经 有 一 


运行 时 库 以 及 对 执行 并 行 代码 的 系统 


些 使 用 


GCD 的 经 历 ， 若 需要 复习 GCD 的 基础 知识 ， 可 以 参阅 Ray Wenderlich 的 “iOS 系统 中 
的 多 线程 和 GCD 的 初学 者 教程 ”(https://www.raywenderlich.com/4295/multithreading-and- 


grand-central-dispatch-on-ios-for- 
性 的 考虑 ， 我 们 快速 看 一 下 GCD 提供 的 功能 列表 。 


并 行 执行 和 串 行 执行 。 
组 任务 执行 情况 的 跟踪 ， 而 与 这 些 任务 所 基于 的 队列 无 关 。 


然而 ， 出 于 完整 
。 任务 或 分 发 队列 ， 
。 分 发 组 ， 实 现 对 一 
。 信号 量 。 
。 屏障 ， 人 允许 在 并 行 
。 分 发 对 象 和 管理 








分 发 队列 




















允许 主线 程 中 的 执行 、 


beginners-tutorial ) 。 


中 创建 同步 的 点 。 


E 源 ， 实 现 更 为 底层 的 管理 和 监控 。 


。 异步 JO， 使 用 文件 描述 符 或 管道 。 





GCD 同样 解决 了 线程 的 创建 与 管理 。 它 帮助 我 们 跟踪 应 用 中 线程 的 总 数 ， 且 不 会 造成 任 


的 泄漏 。 


H 





faf 
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大 多 数 情况 下 ， 应 用 单独 使 用 GCD 就 可 以 很 好 地 工作 ， 但 仍 有 特定 的 情况 
需要 考虑 使 用 NSThread 或 NSOperationQueue。 当 应 用 中 有 多 个 长 耗 时 的 任 
务 需要 并 行 执行 时 ， 最 好 . 对 线程 的 创建 过 程 加 以 控制 。 如 果 代 码 执行 的 时 
间 过 长 ， 很 有 可 能 达到 线程 的 限制 64 个 ，” 即 GCD 的 线程 池上 限 。 

应 该 避免 浪费 地 使 用 dispatch async 和 dispatch_sync， 因 为 那 会 导致 应 用 
崩溃 “。 虽 然 64 个 线程 对 移动 应 用 来 说 是 个 很 高 的 合理 值 ， 但 不 加 控制 的 应 
用 迟早 会 超出 这 个 限制 。 









































4.4 操作 与 队列 


操作 和 操作 队列 是 IOS 编程 中 和 任务 管理 有 关 的 又 一 个 重要 概念 。 


NSOperation 封装 了 一 个 任务 以 及 和 任务 相关 的 数据 和 代码 ， 而 NSOperationQueue 以 先入 
先 出 的 顺序 控制 了 一 个 或 多 个 这 类 任务 的 执行 。 


NSOperation 和 NSOperationQueue 都 提供 控制 线程 个 数 的 能 力 。 可 用 maxConcurrentOpera- 
tionCount 属性 控制 队列 的 个 数 ， 也 可 以 控制 每 个 队列 的 线程 个 数 。 


在 使 用 NSThread (开发 人 员 管理 全 部 并 发 ) 和 GCD (OS 管理 并 发 ) 之 间 存 在 两 个 选择 。 
以 下 是 对 NSThread, NSOperationQueue 和 GCD API 的 一 个 快速 比较 。 


























* GCD 
4 抽象 程度 最 高 。 
4 两 种 队列 开 箱 即 用 : main 和 global。 
e 可 以 创建 更 多 的 队列 (使 用 dispatch queue create), 
e 可 以 请 求 独占 访问 (使 用 dispatch barrier sync 和 dispatch barrier async), 
4 基于 线程 管理 。 
4 硬性 限制 创建 64 个 线程 。 
e NSOperationQueue 
4 无 默认 队列 。 
4 应 用 管理 自己 创建 的 队列 。 
4 队列 是 优先 级 队列 。 
4 操作 可 以 有 不 同 的 优先 级 (使 用 queuePriority 属性 ) 。 
注 2: Stack Overflow, “Number of Threads Created by GCD?” (http://stackoverflow.com/questions/7213845/number- 


ik 3; 


ik 4: 


of-threads-created-by-gcd#0). 

Stack Overflow, “Workaround on the Threads Limit in Grand Central Dispatch?" (http://stackoverflow.com/ 
questions/15150308/workaround-on-the-threads-limit-in-grand-central-dispatch#0). 

Stack Overflow, “GCD Dispatch Concurrent Queue Freeze with ‘Dispatch Thread Soft Limit Reached: 64’ 
in Crash Log" (http://stackoverflow.com/questions/14027824/gcd-dispatch-concurrent-queue-freeze-with- 
dispatch-thread-soft-limit-reached-6#0). 
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* 使 用 cancel 消息 可 以 取消 操作 。 注 意 ，cancel 仅仅 是 个 标记 。 如 果 操 作 已 经 开始 
执行 ， 则 可 能 会 继续 执行 下 去 。 
4 可 以 等 待 某 个 操作 执行 完毕 (EJH wattUntilFinished 消息 )。 
e NSThread 
4 低级 别 构造 ， 最 大 化 控制 。 
应 用 创建 并 管理 线程 。 
应 用 创建 并 管理 线程 池 。 
应 用 启动 线程 。 
线程 可 以 拥有 优先 级 ， 操 作 系 统 会 根据 优先 级 调度 它们 的 执行 。 
无 直接 API 用 于 等 待 线程 完成 。 需 要 使 用 互 斥 量 (如 NSLock) 和 自 定义 代码 。 





* 
* 
* 
* 
* 


NSOperationQueue 是 多 核 安 全 的 。 你 可 以 放心 地 分 享 队列 ， 从 不 同 的 线程 中 
提交 任务 ， 而 无 需 担心 损坏 队列 。 





之 = b 

45 ”线程 安全 的 代码 

贯穿 软件 开发 的 职业 生涯 ， 我 们 总 是 被 教导 要 编写 线程 安全 的 代码 ， 这 也 就 是 说 ， 如 果 有 
多 个 线程 并 行 地 执行 同一 组 指令 ， 不 能 产生 任何 副作用 。 

以 下 两 大 类 技术 可 以 实现 这 一 点 。 


。 不 要 使 用 可 修改 的 共享 状态 。 
。 如 果 无 法 避免 使 用 可 修改 的 共享 状态 ， 则 确保 你 的 代码 是 线程 安全 的 。 


这 些 技术 说 起 来 容易 做 起 来 难 。 要 实现 它们 有 多 种 选择 。 
因为 应 用 会 包含 可 修改 的 共享 状态 ， 所 以 我 们 需要 掌握 管理 和 修改 共享 状态 的 最 佳 实践 。 
驱动 这 些 最 佳 实践 的 一 条 基本 规则 是 “在 代码 中 保留 不 变量 "。” 


4.5.1 原子 属性 

原子 属性 是 实现 应 用 状态 线程 安全 的 一 个 良好 开始 。 如 果 一 个 属性 是 atomic， 则 修改 和 读 
取 肯 定 都 是 原子 的 。 

这 一 点 很 重要 ， 因 为 这 样 可 以 阻止 两 个 线程 同时 更 新 一 个 值 ， 反 之 则 有 可 能 导致 错误 的 状 
态 。 正 在 修改 属性 的 线程 必须 处 理 完毕 后 ， 其 他 线程 才能 开始 处 理 。 
所 有 的 属性 默认 都 是 原子 性 的 。 作 为 最 佳 实践 ， 在 需要 时 应 该 显 式 地 使 用 atomic, AE 
用 nonatomic 标记 属性 。 例 4-2 演示 了 atomic 和 nonatomic 属性 。 




































































iE 5: Stack Overflow, “What Is an Invariant?” (http://stackoverflow.com/a/112088). 
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例 4-2 atomic 和 nonatomic 属性 


@property (atomic) NSString *firstName; @ 
@property (nonatomic) NSString *department; @ 


@ 原子 属性 
@ 非 原 子 属性 


因为 原子 属性 存在 开销 ， 所 以 过 度 使 用 它们 并 不 明智 。 例 如 ， 如 果 能 够 保证 某 个 属 ; 
何 时 刻 都 不 会 被 多 个 线程 访问 ， 那 最 好 还 是 将 其 标记 为 nonatomic, 


fd: FA IBOutlet 时 就 是 一 个 很 好 的 例子 。@property (monatomic, readwrite, strong) 
IBOutlet UILabel *nameLabel 要 比 @property (atomic, readwrite, strong) IBOutlet 
UILabel *nameLabel 更 好 ， 因 为 UIKit 只 允许 在 主线 程 中 操纵 UI 元素 。 由 于 只 会 在 指定 的 
线程 内 进行 访问 ， 除 了 带 来 额外 开销 ， 将 属性 设置 为 atomic 不 会 带 来 任何 价值 。 


4.5.2 同步 块 


即使 属性 被 标记 为 atomic， 最 终 使 用 它们 的 代码 仍 可 能 是 线程 不 安全 的 。 原 子 必 性 只 能 阻 
止 并 行 修改 。 假 设 我 们 有 一 个 HPUser 实体 类 ， 可 以 使 用 HPOperation 对 其 进行 更 新 ， 如 例 
4-3 所 示 。 


例 4-3 在 线程 间 使 用 原子 属性 
// 一 个 实体 (部 分 定义 ) 


@interface HPUser 





= 


PE TELE 












































(property (atomic, copy) NSString *firstName; 
(property (atomic, copy) NSString *lastName; 


(end 





// 一 个 服务 类 (出 于 简洁 的 目的 省 略 了 声明 ) 


(implementation HPUpdaterService 


— 


-(void)updateUser:(HPUser *)user properties:(NSDictionary *)properties { 
NSString *fn = [properties objectForKey:@"firstName" ]; 
if(fn != nil) { 
user.firstName - fn; 


j 


NSString *ln = [properties objectForKey:@"lastName" ]; 
if(ln != nil) { 
user.lastName - ln; 
} 
} 


@end 


每 当 用 户 下 拉 刷 新 且 数 据 从 服务 器 返回 时 ，updateUser:properties: 方法 都 会 被 调用 。 此 
外 ， 一 个 周期 性 执行 的 同步 任务 也 会 调用 该 方法 。 


因此 ， 在 某 个 时 间 点 可 能 会 有 多 个 响应 同时 尝试 更 新 用 户 配 置 文件 一 一 可 能 通过 两 个 CPU 
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核心 或 通过 不 同 的 时 间 片 。 


思考 以 下 场景 : 两 个 响应 在 不 同 线程 中 试图 更 新 用 户 ， 名 称 分 别 为 Bob Taylor 和 Alice 
Darji。 如 果 不 对 属性 FirstName 和 LastName 使 用 原子 更 新 ， 则 无 法 确保 执行 顺序 ， 最 终 的 
结果 可 能 是 任意 组 合 ， 其 中 包括 Alice Taylor 和 Bob Darji。 


这 个 示例 只 是 用 于 演示 ， 其 目的 是 强调 原子 属性 并 不 能 保证 代码 一 定 是 线程 安全 的 。 

所 以 我 们 提供 了 下 一 个 最 佳 实践 : 所 有 相关 的 状态 都 应 该 在 同一 个 事务 中 批量 更 新 。 

使 用 @synchronized 指令 可 以 创建 一 个 信号 量 ， 并 进入 临界 区 ， 临 界 区 在 任何 时 刻 都 只 能 
被 一 个 线程 执行 。 例 4-4 展示 了 改进 后 的 代码 。 


例 4-4 线程 安全 的 块 


(implementation HPUpdaterService 



































-(void)updateUser:(HPUser *)user properties:(NSDictionary *)properties { 
@synchronized(user) { @ 
NSString *fn = [properties objectForKey:@"firstName" ]; 
if(fn != nil) { 
user.firstName = fn; 
} 
NSString *ln = [properties objectForKey:Q"lastName"]; 
if(ln != nil) { 
user.lastName - ln; 
} 
} 
} 


@end 
Q 取得 针对 user 对 象 的 锁 。 一 切 相关 的 修改 都 会 被 一 同 处理 ， 而 不 会 发 生 竞争 状态 。 
经 过 这 样 的 改进 ， 用 户 最 终 的 名 字 只 能 是 Bob Taylor 或 Alice Darji, 




















注意 ， 过 度 使 用 @synchronized 指令 会 拖 慢 应 用 的 运行 速度 ， 因 为 任何 时 间 
都 只 有 一 个 线程 在 临界 区 内 执行 。 




















在 本 例 中 ， 我 们 通过 user 对 象 获取 锁 。 因 此 ，updateUser:properties 方法 可 以 从 多 个 线 
程 被 调用 ， 不 同 的 用 户 用 不 同 的 线程 。 当 user 对 象 不 同时 ， 该 方法 仍然 能 够 高 并 发 地 执 
行 。 结 果 是 代码 既 实 现 了 高 并 发 ， 又 设置 了 警戒 以 防止 数据 冲突 。 








获取 锁 的 对 象 是 良好 定义 的 临界 区 的 关键 。 作 为 经 验 法 则 ， 可 以 选择 状态 会 
被 访问 和 修改 的 对 象 作 为 信号 量 的 引用 。 
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一 切 到 目前 为 止 都 还 好 。 但 是 读 取 属 性 时 应 采取 什么 策略 呢 ? 如 有 果 在 需要 显示 HPUser 对 象 
的 全 名 时 ， 它 正在 被 修改 ， 又 该 如 何 处 理 呢 ? 


























4.5.3 $i 
锁 是 进入 临界 区 的 基础 构件 。atomic 属性 和 @synchronized 块 是 为 了 实现 便捷 实用 的 高 级 
别 抽象 。 
以 下 是 三 种 可 用 的 锁 。 
e NSLock 
这 是 一 种 低级 别 的 锁 。 一 旦 获取 了 锁 ， 执 行 则 进入 临界 区 ， 且 不 会 允许 超过 一 个 线程 并 
行 执行 。 释 放 锁 则 标记 着 临界 区 的 结束 。 


例 4-5 展示 了 如 何 使 用 NSLock。 
例 4-5 使 用 NSLock 


@interface ThreadSafeClass () { 
NSLock *lock; @ 


} 
@end 











-(instancetype)init { 
if(self = [super init]) { 
self->lock = [NSLock new]; © 


return self; 


} 


-(void)safeMethod { 
[self->lock lock]; © 


// 线 程 安全 的 代码 0 


[self->lock unlock]; © 
} 


O 将 锁 声 明 为 一 个 私有 字段 ， 也 可 以 用 属性 来 表示 锁 。 

e 初始 化 锁 。 

© 获取 锁 ， 进 入 临界 区 。 

O 在 临界 区 ， 任 意 时 刻 最 多 只 允许 一 个 线程 执行 。 

O 释放 锁 标 记 着 临界 区 的 结束 。 其 他 线程 现在 能 够 获取 锁 了 。 

NSLock 必须 在 锁定 的 线程 中 进行 解锁 。 
。 NSRecursiveLock 
在 调用 lock 之 前 ，NSLock 必须 先 调 用 unLock。 但 正如 名 字 所 暗示 的 那样 
NSRecursiveLock 克 许 在 被 解锁 前 锁定 多 次 。 如 果 解 锁 的 次 数 与 锁定 的 次 数 相 匹 配 ， 则 
认为 锁 被 释放 ， 其 他 线程 可 以 获取 锁 。 





















































当 类 中 有 多 个 方法 使 用 同一 个 锁 进 行 同步 ， 且 其 中 一 个 方法 调用 另 一 个 方法 时 ， 
NSRecursiveLock 非常 有 用 。 例 4-6 展示 了 使 用 NSRecursiveLock 的 一 个 例子 。 


(ll 4-6 ”使 用 NSRecursiveLock 


@interface ThreadSafeClass () { 
NSRecursiveLock *lock; @ 


} 
@end 


-(instancetype)init { 
if(self = [super init]) { 
self->lock = [NSRecursiveLock new]; 


} 


return self; 


-(void)safeMethod1 { 
[self->lock lock]; @ 


[self safeMethod2]; © 


[self-»lock unlock]; © 
} 


-(void)safeMethod2 { 
[self->lock lock]; @ 


// 线 程 安全 的 代码 


[self->lock unlock]; © 
} 


@ NsRecursivelock 对 象 。 

@ safeMethod1 方法 获取 锁 。 

© 它 调 用 了 safeMethod2 方法 。 

Q safemethod2 从 已 经 获取 到 的 锁 再 次 获取 了 锁 。 

@ safeMethod2 释放 了 锁 。 

Q safeMethod1 释放 了 锁 。 因 为 每 个 锁定 操作 都 有 一 个 相应 的 解锁 操作 与 之 匹配 ， 所 

以 锁 现 在 被 释放 ， 并 可 以 被 其 他 线程 所 获取 。 

NSCondition 
有 些 情况 需要 协调 线程 之 间 的 执行 。 例 如 ， 一 个 线程 可 能 需要 等 待 其 他 线程 返回 结果 。 
NSCondition 可 以 原子 性 地 释放 锁 ， 从 而 使 得 其 他 等 待 的 线程 可 以 获取 锁 ， 而 初始 的 线 
程 继续 等 待 。 
一 个 线程 会 等 待 释放 锁 的 条 件 变量 。 另 一 个 线程 会 通知 条 件 变量 释放 该 锁 ， 并 唤醒 等 待 
中 的 线程 。 
使 用 NSCondition 解决 标准 的 生产 者 一 消费 者 问题 。 例 4-7 中 的 代码 展示 了 如 何 解决 这 


个 问题 。 
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例 4-7 使 用 NSCondition 


@implementation Producer 


- (ánstancetype)initWithCondition:(NSCondition *)condition 
collector:(NSMutableArray *)collector ( @ 
if(self = [super init]) { 
self.condition - condition; 
self.collector - collector; 
self.shouldProduce = NO; 
self.item = nil; 
} 
return self; 


} 


-(void)produce { 
self.shouldProduce = YES; 
while(self.shouldProduce) { @ 
[self.condition lock]; © 
if(self.collector.count > 0) { 
[self.condition wait]; @ 
} 
[self.collector addObject:[self nextItem]]; @ 
[self.condition signal]; © 
[self.condition unlock]; @ 
} 
} 
@end 


@implementation Consumer 


- (ànstancetype)initWithCondition:(NSCondition *)condition 
collector:(NSMutableArray *)collector ( @ 
if(self = [super init]) { 
self.condition - condition; 
self.collector - collector; 
self.shouldConsume = NO; 
self.item = nil; 
} 
return self; 


} 


-(void)consume { 
self.shouldConsume = YES; 
while(self.shouldConsume) { © 
[self.condition lock]; @ 
if(self.collector.count == 0) { 
[self.condition wait]; QD 
} 
id item = [self.collector objectAtIndex:0]; 
// 处 理 产 品 
[self.collector removeItemAtIndex:0]; @ 
[self.condition signal]; (9 
[self.condition unlock]; @ 








} 
@end 


@implementation Coordinator 


-(void)start { 
NSMutableArray *pipeline = [NSMutableArray array]; 
NSCondition *condition = [NSCondition new]; (9 
Producer *p = [Producer initWithCondition:condition 
collector:pipeline]; 
Consumer *c - [Consumer initWithCondition:condition 
collector:pipeline]; (9 
[[NSThread initWithTarget:self selector:QSEL(startProducer) 
object:p] start]; 
[[NSThread initWithTarget:self selector:QSEL(startCollector) 
object:c] start]; @ 
// 一 旦 完成 
p.shouldProduce = NO; 
c.shouldConsume = NO; (9 
[condition broadcst]; (9 
} 


@end 


@O@ 生 产 者 的 初始 化 器 需要 用 于 协调 配合 的 NSCondition 对 象 和 用 于 存放 产品 的 
collector。 初 始 状 态 设 置 为 不 要 生产 (shouldProduce = NO), 

四 生产 者 会 在 shouldProduce 为 YES 时 进行 生产 。 其 他 线程 需要 将 其 设置 为 NO 以 停 
止 生产 者 的 生产 。 

© 获取 condition 的 锁 ， 进 入 临界 区 。 

O 如 果 collector 中 有 未 消费 的 产品 ， 则 等 待 ， 这 会 阻塞 当前 线程 的 执行 直到 
condition 被 通知 (signal) 为 止 。 

© 将 生产 的 nextItem 送 入 collector 以 供 消 费 。 

Q 通知 其 他 等 待 的 线程 (如果 存在 )。 这 里 是 产品 完成 生产 的 标志 ， 并 将 产品 加 入 到 了 
collector 中 ， 可 供 消费 。 

@ 释放 锁 。 

© 消费 者 的 初始 化 器 需要 用 于 协调 配合 的 NSCondition 对 象 和 用 于 存放 产品 的 
collector。 在 初始 化 时 设置 为 不 消费 (shouldConsume = NO), 

Q 当 shouldConsume 75 YES 时， 消费 者 会 进行 消费 。 其 他 线程 可 以 将 其 设置 为 NO 来 停 
止 消 费 者 的 消费 。 

@ 获取 condition 的 锁 ， 进 入 临界 区 。 

@ 如 果 collector 中 没有 产品 ， 则 等 待 。 

@ 消费 collector 中 的 下 一 个 产品 。 确 保 已 经 从 collector 中 移 除 它 。 

© 通知 其 他 等 待 的 线程 (如果 存 在 )。 这 里 标识 一 个 产品 被 消费 并 从 collector 中 移 除了 。 

Q 释放 锁 。 

@ Coordinator 类 为 生产 者 和 消费 者 准备 好 了 输入 数据 (具体 指 的 是 collector 和 condition), 

O 设置 生产 者 和 消费 者 。 

@ 在 不 同 的 线程 中 开启 生产 和 消费 任务 。 
















































































@ 一 量 完 成 ,分 别 设置 生产 者 和 消费 者 停止 生产 和 消费 。 
Q 因为 生产 者 和 消费 者 线程 可 能 会 等 待 ， 所 以 broadcast 本 质 上 会 通知 所 有 等 待 中 的 
线程 。 不 同 的 是 ，signat 方法 只 会 影响 一 个 等 待 的 线程 。 


4.5.4 将 读 写 锁 应 用 于 并 发 读 写 


从 本 节 开始 ， 我 们 将 讨论 能 够 实现 线程 安全 的 两 个 方案 。 本 节 将 介绍 各 免 并 发 写 入 的 最 人 
实践 ， 然 后 在 下 一 节 讨论 不 可 变 实体 。 


我 们 已 经 学 习 了 atomic 属性 可 以 用 于 防护 不 一 致 性 更 新 的 问题 ， 但 有 些 过 于 谨慎 。 如 果 有 
多 个 线程 试图 读 取 一 个 属性 ， 同 步 的 代码 在 同一 时 刻 只 允许 单个 线程 进行 访问 。 因 此 ,使 
用 atomic 属性 会 拖 慢 应 用 的 性 能 。 

这 可 能 是 个 严重 的 瓶 须 ， 尤 其 是 当 某 个 状态 需要 在 多 个 线程 间 共 享 ， 且 需要 被 多 个 线程 访 
问 时 。cookie 或 登录 后 的 访问 令 牌 就 是 这 样 的 例子 。 它 可 以 周期 性 地 变化 ， 但 会 被 所 有 访 
问 服务 器 的 网 络 请 求 所 调用 。 
另外 一 个 使 用 案例 是 丝 存 。 每 条 缓存 的 条 目 可 以 被 应 用 内 的 任何 地 方 所 访问 ， 并 且 会 因为 
用 户 特定 的 操作 而 更 新 。 


本 质 上 ， 我 们 需要 允许 并 行 读 取 、 却 与 写 人 互 斥 的 一 种 机 制 。 这 将 我 们 带 到 了 读 写 锁 的 话 
题 。 它 们 通常 有 多 个 读者 、 单 一 写 者 锁 和 多 个 读者 、 多 个 写 者 锁 。 

读 写 锁 允 许 并 行 访问 只 读 操作 ， 而 写 操作 需要 互 斥 访问 。 这 意味 着 多 个 线程 可 以 并 行 地 读 
取 数据 ， 但 是 修改 数据 时 需要 一 个 互 斥 锁 。 

GCD 屏障 允许 在 并 行 分 发 队列 上 创建 一 个 同步 的 点 。 当 过 到 屏障 时 ，GCD 会 延迟 执行 提 
交 的 代码 块 ， 直 到 队列 中 所 有 在 屏障 之 前 提交 的 代码 块 都 执行 完毕 。 随 后 ， 通 过 屏障 提交 
的 代码 块 会 单独 地 执行 。 我 们 将 这 个 代码 块 称 为 屏障 块 。 待 其 完成 后 ， 队 列 会 按照 原 有 行 
为 继续 执行 。 

图 4-2 演示 了 屏障 在 多 线程 环境 中 执行 的 效果 。 块 1 到 块 6 可 以 在 多 线程 间 并 行 执行 。 但 
是 屏障 块 单独 地 执行 。 唯 一 的 限制 是 ， 所 有 的 执行 都 必须 通过 并 行 队列 进行 执行 。 
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图 4-2: 分 发 块 和 屏障 
要 想 实 现 这 一 行为 ， 我 们 需要 遵循 以 下 步骤 。 
(1) 创建 一 个 并 行 队列 。 








(2) 在 这 个 队列 上 使 用 dispatch_sync 执行 所 有 的 读 操作 。 
(3) 在 相同 的 队列 上 使 用 dispatch barrier sync 执行 所 有 的 写 操作 。 


你 可 以 使 用 例 4-8 中 的 代码 来 实现 高 吞吐 量 且 线程 安全 的 模型 。 
例 4-8 线程 安全 且 高 吞吐 量 的 模型 


//HPCache.h 
@interface HPCache 





+(HPCache *)sharedInstance; 


-(id)objectForKey:(id) key; 
-(void)setObject:(id)object forKey:(id)key; 


Qend 


//HPCache.m 
@interface HPCache () 


@property (nonatomic, readonly) NSMutableDictionary *cacheObjects; 
@property (nonatomic, readonly) dispatch queue t queue; 


@end 
@implementation HPCache 


-(instancetype)init { 
if(self = [super init]) { 
_cacheObjects = [NSMutableDictionary dictionary]; 
_queue = dispatch_queue_create(kCacheQueueName, 
DISPATCH_QUEUE_CONCURRENT); Qj 
} 
return self; 


} 


+(HPCache *)sharedInstance { 
static HPCache *instance = nil; 


static dispatch_once_t onceToken; 
dispatch_once(&onceToken, “{ 

instance = [[HPCache alloc] init]; 
IDE 
return instance; 


} 


- (id )objectForKey: (id<NSCopying>)key { 
. block id rv = nil; 


dispatch sync(self.queue, ^( € 
rv = [self.cacheObjects objectForKey:key]; 
DE 


return rv; 
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-(void)setObject:(id)object forKey:(id«NSCopying»)key { 
dispatch barrier, async(self.queue, ^( © 
[self.cacheObjects setObject:object forKey:key]; 
DH 
} 


@end 


Q 创建 一 个 自 定义 的 DISPATCH_QUEUE_CONCURRENT 队列 , 

@ 将 dispatch_sync (或 dispatch_async) 用 于 不 修改 状态 的 操作 。 

© 将 dispatch_barrier_sync (或 dispatch_barrier_async) 用 于 可 能 修改 状态 的 操作 。 
注意 ， 这 里 的 属性 被 标记 为 nonatomic， 因 为 这 里 有 自 定义 的 代码 使 用 了 自 定义 的 队列 和 
屏障 来 管理 线程 安全 。 


4.5.5 ”使 用 不 可 变 实体 

一 切 看 起 来 都 很 完美 。 但 是 如 果 需 要 访问 一 个 正在 修改 的 状态 ， 那 将 会 怎么 样 呢 ? 

例如 ， 如 果 缓 存 被 清空 ， 但 因为 用 户 执 行 了 一 个 交互 ， 其 中 部 分 状态 要 求 立 即 被 使 用 ， 

情况 将 会 是 怎样 的 呢 ? 是 否 存在 更 有 效 的 机 制 以 管理 状态 ， 而 不 是 多 个 组 件 试图 同时 更 

新 状态 ? 

你 的 团队 应 该 遵循 以 下 的 最 佳 实践 。 

。 使 用 不 可 变 实 体 。 

。 通过 更 新 子 系统 提供 支持 。 

。 多 许 观察 者 接收 有 关 数 据 变化 的 通知 。 

这 就 创建 了 一 个 解 耦 的 、 可 伸缩 的 系统 来 管理 应 用 的 状态 。 我 们 党 试 从 诸多 可 能 的 方案 中 

选取 一 个 予以 实现 。 

首先 要 清晰 地 定义 模型 。 在 研究 案例 中 ， 我 们 定义 了 以 下 三 个 实体 。 

e HPUser 
表示 系统 中 的 一 个 用 户 。 每 个 用 户 有 唯一 的 ID 、 拆 分 为 FirstName 和 LastName 的 姓名 、 
性 别 和 出 生日 期 。 

e HPAlbum 
表示 一 个 相册 。 每 个 用 户 有 0 个 或 多 个 相册 。 每 个 相册 包含 唯一 的 序列 号 、 持 有 者 、 名 
称 、 创 建 时 间 、 描 述 、 封 面 图 片 链接 和 点 赞 (喜欢 该 相册 的 用 户 )。 

e HPPhoto 
表示 相册 内 的 一 张 照片 。 每 个 相册 可 以 包含 0 张 或 多 张 照 片 。 每 张 照片 包含 唯一 的 序列 
号 、 所 属 的 相册 、 用 户 (上 传 照片 的 人 )、 标 题 、url 和 大 小 〈 宽 度 和 高 度 ) 。 


例 4-9 展示 了 定义 实体 的 代码 。 

























































































例 4-9 用 于 案例 研究 的 实体 ， 表 示 用 户 、 相 册 和 照片 


(interface HPUser 


(property (nonatomic, copy) NSString *userId; 
(property (nonatomic, copy) NSString *firstName; 
(property (nonatomic, copy) NSString *lastName; 
(property (nonatomic, copy) NSString *gender; 
(property (nonatomic, copy) NSDate *dateOfBirth; 
(property (nonatomic, strong) NSArray *albums; 


Qend 
@class HPPhoto; 
@interface HPAlbum 


@property (nonatomic, copy) NSString *albumId; 
@property (nonatomic, strong) HPUser *owner; 
@property (nonatomic, copy) NSString *name; 
@property (nonatomic, copy) NSString *description; 
@property (nonatomic, copy) NSDate *creationTime; 
@property (nonatomic, copy) HPPhoto *coverPhoto; 


Qend 
(interface HPPhoto 


(property (nonatomic, copy) NSString *photold; 
Gproperty (nonatomic, strong) HPAlbum *album 
@property (nonatomic, strong) HPUser *user; 
(property (nonatomic, copy) NSString *caption; 
(property (nonatomic, strong) NSURL *url; 
(property (nonatomic, copy) CGSize size; 


@end 
有 多 种 方式 可 以 定义 填充 数据 的 模型 和 机 制 。 甚 中 两 个 常见 的 方案 是 : 
。 使 用 自 定义 的 初始 化 器 
。 使 用 生成 器 模式 
每 个 方案 各 有 优点 。 
使 用 自 定义 的 初始 化 器 意味 着 会 有 很 长 的 方法 名 ， 从 而 导致 令 人 讨厌 的 调用 ， 比 如 方法 in 


itwithId:firstName:lastName:gender:birthday:。 况 且 这 只 是 我 们 使 用 模型 中 部 分 属性 的 
青 况 。 如 有 果 再 加 五 个 属性 ， 初 始 化 器 会 迅速 膨胀 。 


Ee a et LN A MM e TIUS edn avs dE 
这 也 令 使 用 了 新 版 模型 的 应 用 能 够 在 编译 时 知道 什么 地 方 发 生 了 改变 


使 用 生成 器 模式 需要 引入 外 部 类 进行 管理 。 生 成 器 有 setter 方法 ， 也 需要 提供 与 模型 数据 
完全 一 致 的 存储 。 生 成 器 最 终 也 会 使 用 初始 化 器 
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模型 的 任何 更 新 都 需要 对 生成 器 及 其 背后 的 属性 做 出 相应 的 改动 。 

推荐 使 用 生成 器 模式 ， 因 为 它 支 持 向 下 兼容 ， 而 且 即 使 不 再 加 入 新 的 属性 ， 生 成 器 也 不 会 
破坏 应 用 。 新 版 模型 中 的 新 增 属性 将 继续 持 有 其 默认 值 。 

第 二 个 方案 中 的 对 应 代码 与 例 4-10 类 似 。 这 部 分 代码 是 基于 Klaas Pieter 使 用 块 (http:// 
www.annema.me/the-builder-pattern-in-objective-c) 实现 生成 器 模式 的 想法 而 演变 的 。 


例 4-10 使 用 生成 器 实现 的 不 可 变 实 体 


//HPUser.h 
@interface HPUserBuilder Qj 




















(property (nonatomic, copy) NSString *userId; 
(property (nonatomic, copy) NSString *firstName; 
(property (nonatomic, copy) NSString *lastName; 
(property (nonatomic, copy) NSString *gender; 
(property (nonatomic, copy) NSDate *dateOfBirth; 
(property (nonatomic, strong) NSArray *albums; 


-(HPUser *)build; 
Qend 
@interface HPUser @ 


// 属 性 





*(instancetype) userWithBlock:(void (^)(HPUserBuilder *))block; 
(gend 

@interface HPUser () © 

-(instancetype) initWithBuilder:(HPUserBuilder *)builder; 


Qend 


(implementation HPUserBuilder 


-(HPUser *) build ( @ 
return [[HPUser alloc] initWithBuilder:self]; 


} 
@end 


@implementation HPUser 
-(instancetype) initWithBuilder:(HPUserBuilder *)builder { © 


if(self = [super init]) { 
self.userId = builder.userId; 
self.firstName - builder.firstName; 
self.lastName - builder.lastName; 
self.gender - builder.gender; 





self.dateOfBirth - builder.dateOfBirth; 
self.albums - [NSArray arrayWithArray:albums]; 
} 


return self; 


} 


+(instancetype) userWithBlock:(void (^)(HPUserBuilder *))block ( © 
HPUserBuilder *builder - [[HPUserBuilder alloc] init]; 


block(builder); 
return [builder build]; 


} 


@end 





// 构 建 对 象 ,一 个 例子 
-(HPUser *) createUser ( @ 


HPUser *rv = [HPUser userWithBlock:^(HPUserBuilder *builder) { 


builder.userId = @"id001"; 
builder.firstName = Q"Alice"; 
builder. LastName = Q"Darji"; 
builder.gender = Q"F"; 


NSCalendar *cal = [NSCalendar currentCalendar]; 


NSDateComponents *components = [[NSDateComponents alloc] init]; 


[components setYear:1980]; 
[components setMonth:1]; 
[components setDay:1]; 


builder.dateOfBirth = [cal dateFromComponents:components]; 


builder.albums = [NSArray array]; 


1 
return rv; 
} 
Q 生成 器 。 


@ 模型 提供 了 类 方法 userwithBlock:, fil 4-9 包含 了 声明 的 全 部 属性 。 


O 模型 的 私有 扩展 一 一 自 定义 的 初始 化 器 。 
O build 方法 的 实现 。 

Q 模型 自 定义 初始 化 器 的 实现 。 

@ userWithBlock: 方法 的 实现 。 

O 用 生成 器 创建 对 象 的 使 用 示例 。 


注意 ， 前 面 的 代码 具有 以 下 优点 。 




















。 模型 总 是 向 下 兼容 。 新 版 的 模型 生成 器 包含 了 新 增 属性 ,但 不 会 破坏 createUser 的 代码 。 
并 调用 build 方法 创建 模型 





。 生成 器 可 以 被 直接 创建 。 模 型 的 消费 者 可 以 初始 化 生成 器 ， 


对 象 。 


。 生成 器 的 创建 和 处 理 可 以 留 给 内 部 核心 完成 。 模 型 的 消费 者 可 以 使 用 类 方法 userwithBlock: 


而 无 需 初始 化 或 亲自 调用 build 方法 。 
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4.5.6 ”使 用 集中 的 状态 更 新 服务 


下 一 步 我 们 需要 一 个 更 新 服务 ， 以 更 新 客户 端 状态 。 更 新 服务 可 能 需要 连接 服务 器 ， 在 执 
行 本 地 更 新 前 进行 验证 ， 如 加 入 或 更 新 一 条 记录 、 确 认 好 友 的 申请 或 上 传 一 张 照片 。 从 UI 
的 视角 来 看 ， 在 短暂 的 期 间 内 ， 你 需要 向 用 户 展示 一 个 进度 条 或 其 他 的 指示 器 ， 以 通知 用 
户 有 关 状 态 的 实际 情况 。 
在 案例 中 ， 我 们 将 HPUserService, HPAlbumService, HPPhotoService 分 别 用 于 服务 HPUser、 
HPAlbum, HPPhoto 对 象 。 


更 新 状态 很 棘手 ， 因 为 状态 是 不 可 变 的 。 有 些 矛 盾 ， 不 是 吗 ? 解决 办 法 是 让 状态 的 生成 器 
接收 一 个 后 续 可 修改 的 输入 状态 。 


为 了 对 HPUser 实现 这 一 点 ， 我 们 可 以 在 HPUserBuilder 上 创建 一 个 辅助 初始 化 器 ， 以 便 接 
收 输入 的 对 象 。 


例 4-11 中 的 代码 展示 了 改进 后 的 HPUserBuilder 类 ， 以 便 支持 对 之 前 创建 的 HPUser x 
进行 修改 ， 同 时 还 创建 了 HPUserService 类 来 获取 和 更 新 对 象 。 类 似 的 基础 改造 也 适用 于 
HPALbum 和 HPPhoto 实体 。 这 段 代码 演示 了 用 于 用 户 和 相册 实体 的 服务 ， 通 常用 于 以 下 两 个 
场景 

。 从 服务 器 获取 数据 并 导致 本 地 状态 的 更 新 ， 

。 更 新 本 地 和 远程 的 状态 ， 例 如 ， 通 过 用 户 的 交互 。 


例 4-11 用 于 用 户 和 相册 对 象 的 服务 


/ | HPUserBuilder.h 
(interface HPUserBuilder 





























-(instancetype) initWithUser:(HPUser *)user; 
Qend 
@interface HPUserBuilder 


-(instancetype) initwithUser:(HPUser *)user { @ 
if(self = [super init]) { 

self.userId - builder.userId; 
self.firstName - user.firstName; 
self.lastName - user.lastName; 
self.gender - user.gender; 
self.dateOfBirth - user.dateOfBirth; 
self.albums - user.albums; 


return self; 


} 
@end 


//HPUserService.h 
@interface HPUserService 





*(instancetype)sharedInstance; @ 
-(void)userWithId:(NSString *)id completion:(void (^)(HPUser *))completion; 
-(void)updateUser:(HPUser *)user completion:(void (^)(HPUser *))completion; 


Qend 


//HPUserService.m 
@interface HPUserService 


@property (nonatomic, strong) NSMutableDictionary *userCache; © 
@end 
@implementation HPUserService 


-(instancetype) init { @ 
if(self = [super init]) { 
self.userCache = [NSMutableDictionary dictionary]; 
} 


return self; 


} 


-(void)userWithId:(NSString *)id completion:(void (^)(HPUser *))completion { @ 
// 检 查 本 地 缓存 或 从 服务 器 提取 
HPUser *user = (HPUser *)[self.userCache objectForKey:id]; 
if(user) { 
completion(user); 


} 


[[HPSyncService sharedInstance] fetchType:@"user" 
withId:id completion:^(NSDictionary *data) { © 
//(8 FAHPUserBuilder, 4r ROE HAIE 
HPUser *userFromServer = [builder build]; 
[self.userCache setObject:userFromServer forKey:userFromServer.userId]; 
callback(userFromServer); 

H; 

} 


-(void)updateUser:(HPUser *)user completion:(void (^)(HPUser *))completion ( @ 
// 可 能 会 要 求 更 新 到 服务 器 
[[HPSyncService sharedInstance] updateType:@"user" 


// 使 用 HPUserButLder ,分 析 数 据 并 构建 
HPUser *updatedUser = [builder build]; 


[self.userCache setObject:updatedUser forKey:updatedUser.userId]; © 
[HPAlbumService updateAlbums:updatedUser.albums]; €) 
completion(updatedUser); 


1 
} 


@end 


@ HPUserBuilder 现在 有 了 另 一 个 自 定义 初始 化 器 。 它 以 HPuser 对 象 作为 输入 参数 ， 并 利 
JH user 对 象 的 值 对 自身 进行 初始 化 。 可 以 通过 属性 的 setter 方法 修改 状态 ， 并 最 终 使 用 
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build 方法 构造 新 的 对 象 。 注 意 ， 虽 然 状态 被 修改 ， 但 旧 的 对 象 却 没 有 发 生 任何 改动 。 
这 也 意味 着 ， 如 果 旧 的 对 象 正 被 其 他 实体 〈 如 视图 控制 器 ) 所 使 用 ， 则 需要 进行 替换 。 
我 们 将 在 下 一 节 探 讨 状 态 变更 的 通知 。 

Q HPUserService 遵循 了 单 例 模式 ， 并 可 以 使 用 sharedInstance 方法 。 出 于 简洁 的 目的 省 
略 了 代码 ， 但 我 们 知道 该 如 何 实现 良好 、 安 全 的 单 例 。 通 和 常 而 言 ， 在 实体 或 服务 层次 使 
用 单 例 并 非 明 智 的 选择 ， 因 为 这 会 带 来 紧密 的 耦合 并 影响 mock 框架 的 使 用 。 此 时 ， 使 
用 可 配置 的 工厂 要 优 于 使 用 单 例 。 工 厂 可 以 创建 可 销毁 的 单 例 。 我 们 将 在 第 10 章 回 顾 
这 一 话题 。 

Qe 作为 快速 原型 ， 服 务 还 持 有 了 自己 所 创建 的 用 户 对 象 的 缓存 。 但 是 ， 将 状态 和 缓存 逻辑 
混在 一 起 并 不 是 个 好 主意 。 应 当 总 是 保持 状态 与 任何 其 他 聪明 的 代码 分 离 。 你 可 能 想 让 
模型 越 春 越 好 。 

@ HPUserservice 初始 化 器 被 复写 来 初始 化 缓存 。 这 只 是 个 权宜 之 计 ， 因 为 我 们 的 目的 是 
集中 讨论 不 变 对 象 如 何 才 能 比 可 变 对 象 更 好 地 服务 应 用 ， 而 后 者 的 状态 可 在 应 用 的 不 同 
地 方 修 改 。 在 实际 的 应 用 中 ， 服 务 对 象 会 访问 状态 ， 后 者 可 以 作为 任何 处 理 或 更 新 的 输 
和 入， 此外， 服务 对 象 还 会 访问 网 络 操作 以 保持 与 服务 器 的 同步 。 

© 使 用 userWithId:completion: 方法 可 以 获取 有 给 定 id 的 用 户 。 如 果 该 对 象 存 在 于 本 地 
状态 ， 则 会 被 返回 。 否 则 ， 它 会 请 求 服 务 器 获取 详细 信息 。 一 旦 完成 ，comptLetion 回调 
会 被 触发 ， 以 通知 调用 者 对 象 可 用 。 

Q 假设 这 里 存在 一 个 同步 服务 HPSyncService。 该 服务 会 从 服务 器 获取 数据 。 还 假设 服务 
器 发 送 了 JSON 对 象 “, 并 被 反 序列 化 为 NSDictionary 对 象 。 抽 取 属 性 和 填充 生成 器 的 代 
码 被 忽略 了 。 一 旦 数据 可 用 ， 我 们 还 将 更 新 本 地 的 缓存 ， 以 避免 后 续 的 服务 器 请 求 。 

@ 使 用 updateUser:completion: 方法 可 以 更 新 用 户 的 状态 。 

Q 更 新 本 地 状态 可 能 需要 同步 修改 服务 器 。 

© 一 旦 通知 了 服务 器 ， 本 地 缓存 就 要 被 更 新 。 因 为 用 户 对 象 持 有 相册 ， 所 以 相册 服务 也 用 
于 更 新 相关 的 相册 。 上 有 具体 来 讲 ， 关 联 的 持 有 者 对 象 要 指向 更 新 后 的 用 户 对 象 。 旧 的 用 户 
对 象 需要 被 析 构 。 注 意 ， 这 里 提供 的 解决 方案 伸缩 性 较 差 . 如 果 其 他 实体 对 象 需要 更 新 
自身 ， 那 该 怎么 办 呢 ? 接 下 来 我 们 将 会 解决 这 个 问题 。 

实体 之 间 的 交叉 引用 是 一 个 需要 注意 的 点 。 用 户 有 一 个 相册 列表 ， 每 个 相册 都 有 一 个 主 

人 。 类 似 地 ， 相 册 中 有 照片 列表 ， 而 每 个 照片 都 有 它 所 隶属 的 相册 。 我 们 其 至 还 没有 为 照 

片 的 评论 建 模 ， 评 论 会 包含 写 评论 的 上 时间、 内 容 和 写 评论 的 作者 。 

不 管 它们 是 强 还 是 弱 ， 创 建 包 含 此 类 交叉 引用 的 不 可 变 对 象 被 刻意 忽略 了 。 我 们 需要 用 户 

对 象 早 于 相册 创建 完成 ， 反 之 亦 然 。 这 是 个 矛盾 的 局 面 。 

解决 之 道 是 ， 如 果 没 有 具体 标记 为 不 可 变 ， 则 保持 对 象 可 变 。 这 就 是 所 谓 的 冰棒 不 变性 。” 

要 想 实现 这 一 点 ， 你 需要 一 个 特殊 的 方法 freeze 或 markImmutable。 为 了 能 够 使 用 这 一 结 

构 ， 你 需要 自 定义 的 setter 方法 ， 以 便 在 允许 修改 前 检查 对 象 是 否 为 不 可 变 。 


现在 我 们 可 以 解决 这 个 死 锁 。 首 先 ， 让 HPAlbum 在 设置 持 有 者 之 前 允许 被 修改 。 我 们 创建 




















































































































注 6: 你 可 能 还 想 考虑 其 他 格式 ， 如 Protobuf, Thrift 或 Avro. 
注 7: Stack Overflow: “How to Design an Immutable Object with Complex Initialization” (http://bit.ly/1FuVRGI) 。 
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HPUser 对 象 ， 并 设置 HPALbum 对 象 的 持 有 者 。 随 后 ， 我 们 调用 HPAlbum 对 象 的 freeze Fy 
法 。 当 全 部 相册 创建 完成 之 后 ， 我 们 将 其 设置 为 HPuser 对 象 的 albums 属性 。 最 终 ， 我 们 
调用 HPUser 对 象 的 freeze 方法 。 

代码 效果 如 例 4-12 所 示 。HPUser 被 更 新 为 拥有 读 / 写 属性 ， 并 在 标记 为 不 可 变 之 前 是 可 变 
的 。 而 且 可 以 设想 的 是 ， 绝 大 部 分 的 使 用 场景 都 不 再 需要 生成 器 了 ， 因 为 属性 本 身 是 读 / 
写 属 性 。 


例 4-12 冰棒 不 可 变 的 实体 


//HPUser.h 
@interface HPUser 





























@property (nonatomic, copy) NSString *userId; @ 
@property (nonatomic, copy) NSString *firstName; 
-(void) freeze; @ 


Qend 


//HPUser.m 
@interface HPUser () 


(property (nonatomic, copy) BOOL frozen; €) 
Qend 
(implementation HPUser 


(synthesize userId = userId; @ 
@synthesize firstName = _firstName; 


-(void) freeze { © 
self.frozen = YES; 


J 


-(void) setUserId:(NSString *)userId { © 
if(!self.frozen) { 
self-&gt; userId - userId; 
} 
} 


-(void) setFirstName:(NSString *)firstName { 
if(!self.frozen) { 
self-&gt; firstName = firstName; 


} 





[fess 省 略 了 其 他 的 setter 


Qend 





// 创 建 对象 
-(HPUser *)sampLeUser { @ 
HPUser *user = [[HPUser alloc] init]; 
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user.userId = @"user-1"; 
user.firstName = @"Bob"; 
user. LastName = @"Taylor"; 
user.gender = Q"M"; 


HPAlbum *album1 = [[HPAlbum alloc] init]; 
albumi.owner = user; @ 

albumi.name = ("Album 1"; 

//…… 其 他 属性 
[album1 freeze]; © 

















HPAlbum *album2 = [[HPAlbum alloc] init]; 
album2.owner = user; 

album2.name = ("Album 2"; 

//…… 其 他 属性 
[album2 freeze]; (9 

















user.albums - [NSArray arrayWithObjects:albumi, album2, nil]; 
[user freeze]; QD 


return user; 


j 


Q 属性 不 再 是 readonly, mæ readwrite (fast), 

Q 我 们 加 入 了 freeze 方法 ， 将 对 象 标记 为 不 可 变 。 对 象 默 认 是 可 变 的 。 

© 一 个 用 于 跟踪 对 象 的 不 可 变 状态 的 标签 。 

O 因为 将 编写 自 定义 的 setter 方法 ， 所 有 我 们 需要 esynthesize 并 告诉 编译 器 使 用 隐藏 的 iVar。 

Q freeze 方法 的 实现 标记 对 象 为 不 可 变 。 

Q 自 定义 setter。 首 先 检 查 对 象 是 否 为 可 变 。 如 果 是 ， 则 更 新 。 如 果 不 是 ， 则 不 更 新 。 你 
还 可 以 在 开发 阶段 抛 出 一 个 异常 ， 来 保障 合法 的 调用 和 识别 任何 错误 的 代码 。 

@ 演示 新 API 使 用 的 示例 代码 。 

Q 将 一 个 用 户 设 置 为 相册 的 持 有 者 。 此 时 ， 两 个 对 象 都 是 可 变 类 型 的 。 

© 将 HPALbum 对 象 标 记 为 不 可 变 。 

O 将 HPUser 对 象 标记 为 不 可 变 。 注 意 ， 只 有 在 这 之 前 才 可 以 使 用 不 可 变 的 相册 对 象 。 


虽然 对 象 在 短 时 间 内 可 以 被 修改 ， 但 我 们 能 够 确保 可 变性 是 短暂 的 ， 且 仅 限 于 创建 对 象 的 
线程 。 在 将 对 象 从 创建 方法 送 入 需要 它们 的 应 用 状态 之 前 ， 你 必须 确保 它们 已 经 被 标记 为 
不 可 变 。 


4.5.7 ”状态 观察 者 与 通知 


上 一 广 留 给 我 们 一 个 未 解决 的 问题 ， 如 果 对 象 被 更 新 ， 那 么 我 们 应 该 如 何 同步 更 新 依赖 ? 
或 者 换个 角度 ， 追 踪 状 态 变 更 的 最 佳 实践 是 什么 ? 


要 跟踪 变更 ， 你 有 如 下 选择 : 
。 键 - 值 观察 



































。 通知 中 心 
。 自 定 义 的 方案 





我 们 在 第 2 章 中 介绍 过 前 两 个 方案 。 键 — 值 观察 非常 适合 用 来 跟踪 对 象 的 属性 。 但 是 在 我 
们 的 方案 中 ， 因 为 对 象 是 不 可 变 ， 而 且 我 们 要 替换 整个 对 象 ， 所 以 键 — 值 观察 没有 用 武之 
地 。 因 此 ， 观 察 者 不 会 收 到 任何 的 回调 通知 。 

通知 中 心 是 优秀 的 解决 方案 。 它 提供 了 实用 的 功能 ， 可 以 满足 大 多 数 的 情况 。 但 问题 在 于 
它 最 终 会 放大 应 用 的 复杂 性 。 例 如 ， 按 相册 标识 号 码 过 滤 更 新 通知 或 尽 可 能 直接 地 将 变化 
抛 向 UL 

这 就 需要 自 定义 的 解决 方案 出 场 了 。 要 实现 这 个 目标 ， 我 们 将 切换 到 响应 式 编程 的 风格 。 
响应 式 编程 是 基于 异步 数据 流 的 编程 方式 。’ 流 是 廉价 且 无 处 不 在 的 ,一 切 都 可 以 是 流 : 变 
量 、 用 户 输入 、 必 性、 缓存 、 数 据 结构 ， 等 等 。 

ReactiveCocoa 库 (https://github.com/ReactiveCocoa/ReactiveCocoa) 实现 了 在 Objective-C 
中 进行 响应 式 编 程 。 它 不 仅 可 以 实现 对 任意 状态 的 观察 ， 还 提供 了 高 级 的 分 类 扩展 ， 以 便 
同步 更 新 UL 元素 (如 UILabeL) 或 响应 视图 的 交互 (如 UIButton)。 





















































函数 响应 式 编程 与 ReactiveCocoa 
应 用 通常 会 消费 、 制 造 和 更 新 数据 。 响 应 式 编程 是 一 种 编程 范式 ， 能 够 表示 数据 流 ， 
而 无 需 担 心 副作用 或 对 其 他 并 发 执行 的 任务 造成 影响 。 
响应 式 编程 背 后 的 核心 思想 是 随时 间 流 逝 体 现 数据 的 值 。 使 用 动态 值 的 数据 流 会 导致 
这 些 值 随 着 时 间 而 变化 。 
肠 数 响应 式 编程 (functional reactive programming, FRP) 允许 响应 式 编程 使 用 内 建 的 
块 方法 进行 函数 式 编程 ， 如 map、reduce、filter、merge， 等 等 。 
ReactiveCocoa 的 灵感 来 自 FERP。 因 为 多 组 件 的 应 用 以 高 内 聚 的 方式 工作 ， 所 以 组 件 状 
态 的 使 用 和 更 新 就 有 了 很 强 的 关联 。 正 因为 这 一 点 ， 创 建 一 个 低 耦 合 、 高 内 聚 的 响应 
式 系统 才 显 得 尤为 重要 。 











我 们 将 使 用 ReactiveCocoa 通知 观察 者 模型 发 生 了 变化 。 观 察 者 可 以 创建 在 任何 地 方 。 


为 了 说 明 目 的 ， 我 们 将 在 每 个 用 户 对 象 的 创建 和 更 新 操作 期 间 添加 通知 。 我 们 也 会 为 相册 
服务 添加 观察 者 ， 以 监视 相册 主人 (用户) 的 变化 。 出 于 完整 性 的 目的 ， 在 UI 中 为 用 户 
监控 相册 列表 的 变化 。 例 4-13 展示 了 代码 的 相关 部 分 。 


例 4-13 观察 者 和 通知 


//HPUserService.m 
-(RACSignal *)signalForUserWithId:(NSString *)id ( @ 
(Qweakify(self); 
return [RACSignal 
createSignal:^RACDisposable *(id<RACSubscriber> subscriber) ( @ 
@strongify(self); 











iE 8: The introduction to Reactive Programming you’ ve been missing (https://gist.github.com/staltz/868e7e9bc2a7 





b8c1f754#reactive-programming-is-programming-with-asynchronous-data-streams). 
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HPUser *userFromCache = [self.userCache objectForKey:id]; 
if(userFromCache) { 
[subscriber sendNext:userFromCache]; 
[subscriber sendCompleted]; 
) else { 
// 假 设 HPSyncService 也 遵循 FRP 风 格 
[[[HPSyncService sharedInstance] 
loadType:Q"user" withId:id] 
subscribeNext:^(HPUser *userFromServer) { 
// 也 更 新 本 地 缓存 和 通知 
[subscriber sendNext:userFromServer]; 
[subscriber sendCompleted]; 
} error: ^(NSError *error) { 
[subscriber sendError:error]; 




















1; 
} 


return nil; 
H 
} 


-(RACSignal *)signalForUpdateUser:(HPUser *)user { © 
(Qweakify(self); 
return [RACSignal 
createSignal:^RACDisposable *(id«RACSubscriber» subscriber) ( @ 
// 更 新 服务 器 
[L[HPSyncService sharedInstance] 
updateType:@"user" withId:user.userId value:user] 
subscribeNext:^(NSDictionary *data) { 
/ [fii FAHPUserBuilder ,分 析 数 据 并 构建 
HPUser *updatedUser = [builder build]; 

















@strongify(self); 
var oldUser = [self.userCache objectForKey:updatedUser .userId]; 
[self.userCache setObject:updatedUser forKey:updatedUser.userId]; 
[subscriber sendNext:updatedUser]; 
[subscriber sendCompleted]; 
[self notifyCacheUpdatedWithUser:updatedUser old:oldUser]; G 

} error: ^(NSError *error) { 
[subscriber sendError:error]; 

}]; 

}]; 
} 


-(void)notifyCacheUpdatedWithUser:(HPUser *)user old:(HPUser *)oldUser { @ 
NSDictionary *tuple = { 
@"old": oldUser, 
@"new": user 
F; 
[NSNotificationCenter .defaultCenter 


postNotificationName:Q"userUpdated" object:tuple]; @ 
j 


-(RACSignal *)signalForUserUpdates:(id)object ( G 
return [[NSNotificationCenter.defaultCenter 
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rac addObserverForName:("userUpdated" object:object] © 
flattenMap:^(NSNotification *note) { 
return note.object; 
H; 
} 


// 在 应 用 的 其 他 地 方 
-(void)retrieveAUser:(NSString *)userId ( @ 
[[[HPUserService sharedInstance] 
signalForUserWithId:userId] 
subscribeNext:^(HPUser *user) ( @ 
// 处 理 用 户 , 或 者 更 新 
) error:^(NSError *) ( 
// 向 用 户 显示 错误 
}]; 




















} 


-(void)updateAUser:(HPUser *)user { @ 
[[[HPUserService sharedInstance] 
signalForUpdateUser:user] 
subscribeNext:^(HPUser *user) ( (9 
// 处 理 用 户 , 或 者 更 新 
) error:^(NSError *) ( 
// 向 用 户 显示 错误 
)]; 

















} 


// 监 听 用 户 更 新 
-watchForUserUpdates { (D 
[[[HPUserService sharedInstance] 
signalForUserUpdates:self] (9 
subcribeNext:^(NSDictionary *tuple) ( (9 
// 用 值 做 一 些 事 情 
HPUser *oldUser objectForKey:@"old"; 
HPUser *newUser objectForKey:@"new"; 








1 
} 


@ signalForUserWithId 方法 并 未 将 块 作为 参数 ， 而 是 返回 了 一 个 可 以 被 链 式 调 用 的 
Promise。 这 里 还 使 用 了 2.15 节 中 介绍 的 Qweakify 和 @strongify 两 个 宏 。 

@ 信 号 的 代码 与 userwithId 中 原来 的 代码 格外 相似 ， 不同 之 处 是 这 里 使 用 了 
RACSubscriber 和 一 个 信号。 
假设 HPSyncService 类 中 的 loadType:withId 方法 也 返回 一 个 信号 ， 即 一 个 RACSignal, 

© signalForUpdateUser 方法 更 新 一 个 HPUser 对 象 。 

O 这 里 创建 了 RACSignal, 

O 当 用 户 被 更 新 ， 你 不 仅 需要 通知 直接 订阅 者 ， 还 要 通知 观察 者 更 新 缓存 。 

Q notifyCacheUpdatedWithUser:old: 广播 了 用 户 对 象 的 变化 。 

@ 这 里 使 用 NSNotificationCenter 是 为 了 简化 。 这 个 方法 不 宜 暴 露 给 HPUserService 类 的 
用 户 。 这 是 个 扩展 方法 。 

Q (在 HPUserService.h 文件 中 的 ) 公开 的 方法 是 signalForUserUpdates: 。 
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© ix B [i JH T ReactiveCocoa 框架 提供 的 分 类 扩展 rac_addObserverForName, PAYT [RJ 
userUpdated 通知 。 它 也 会 实际 从 NSNotification 中 抽取 NSDictionary, Jn FH IH HJ 
新 的 用 户 对 象 所 组 成 。 

@ retrieveAUser: 方法 演示 了 获取 用 户 的 示例 代码 。 

@ subscribeNext: 块 是 接收 user 对 象 的 地 方 。 

Q9 updateAUser: 方法 演示 了 更 新 用 户 的 示例 代码 。 

(9 subscribeNext: 块 是 接收 user 对 象 的 地 方 。 

@ watchForUserUpdates: 方法 展示 了 在 用 户 缓存 中 观察 变化 的 示例 代码 。 

Q 它 使 用 signalForuserUpdates: 方法 监听 用 户 缓存 发 生 改 变 的 通知 。 

(9 subscribeNext: 块 会 传人 包含 新 旧 用 户 对 象 的 NSDictionary, 


这 里 的 优点 在 于 ， 如 果 未 来 signalForUserUpdates: 的 实现 不 再 依赖 NSNotificattonCenter ， 
那 将 不 会 影响 watchForUserUpdates: 方法 的 实现 方案 。 


使 用 这 个 库 的 主要 动机 是 ， 它 可 以 帮助 我 们 实现 一 个 观察 变化 的 系统 ， 这 个 系统 有 着 
低 耦 合 、 高 伸缩 性 、 自 包含 且 适 用 于 通用 目的 的 特点 。 更 重要 的 是 ， 它 为 链接 (使 用 
RACSignal) 提供 了 Promise， 使 得 我 们 能 够 写 出 更 易 理解 和 维护 的 代码 。 它 还 提供 了 更 加 
简便 的 方案 以 实现 与 UL 元 素 的 交互 ， 有 些 元 素 会 在 后 续 章 节 中 体现 出 来 。 简 而 言 之 ， 它 
提供 了 许多 样本 代码 ， 没 有 它 我 们 就 得 自己 编写 这 些 代码 ， 而 且 要 做 得 更 多 。 





















































使 Facebook 的 新 闻 流 在 iOS 上 加 速 5096 
2012 年 ，Facebook 将 其 新 闻 流 从 HTML 迁移 到 了 原生 的 iOS 应 用 以 优化 性 能 。 但 是 
久而久之 ， 随 着 包括 小 组 、 主 页 和 时 间 轴 的 其 他 部 分 都 迁移 到 了 原生 地 ， 新 闻 流 拉 低 
了 整体 的 性 能 。 分 析 诊 断 显示 根本 原因 在 数据 层 。?” 
因此 ， 模 型 层 基 于 三 个 原则 进行 了 重 写 : 
。 不 可 变性 
。 非 规范 化 存储 
€ 异步 ， 选 择 性 一 致 性 
这 也 意味 着 放弃 了 Core Data 框架 ， 该 框架 能 保证 很 强 的 数据 一 致 性， 但 却 会 导致 性 能 
AX, 











45.8 异步 优 于 同步 


在 上 一 节 中 ， 我 们 了 解 了 应 该 优先 选用 Promise。 本 节 提 供 了 一 些 针对 异步 代码 的 更 深入 
讨论 。 


一 个 有 说 服 力 且 令 人 印象 深刻 的 理由 支持 我 们 首选 异步 方式 而 不 是 同步 方式 。 且 这 与 同步 











iE 9: Facebook, “Making News Feed Nearly 50% Faster on iOS" (http://bit.ly/faster--ios). 
ik 10: “Facebook’s iOS Architecture - @Scale 2014 - Mobile" (). 
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AK. RIE 45 节 中 讨论 了 使 用 屏障 ， 并 学 习 了 如 何 将 dtspatch_sync 用 于 并 发 读 取 。 
接 下 来 我 们 简单 分 析 一 下 例 4-14 中 的 代码 。 
例 4-14 在 实际 场景 中 使 用 dispatch sync 


d 








// 场 景 A 
dispatch sync(queue, ^() { 
dispatch sync(queue, ^() ( 
NSLog(@"nested sync call"); 
IDE 
IDE 


// 场 景 B 
-(void) methodA1 { 
dispatch_sync(queue1, ^() { 
[objB methodB]; 
DE 
} 


-(void)methodA2 { 
dispatch_sync(queue1, ^() { 
NSLog(@"indirect nested dispatch_sync"); 
}) 
} 


-(void) methodB { 
[objA methodA2]; 


} 





在 例 4-14, 场景 A 演示 了 一 个 假设 的 场景 ， 在 这 个 场景 中 使 用 分 发 队列 调用 了 一 个 嵌 套 的 





ispatch sync, 3X E SUB, PEAY dispatch sync 不 能 分 发 到 队列 中 ， 因 为 当前 线 


程 已 经 在 队列 中 且 不 会 释放 锁 。 

场景 B 演示 了 更 相似 的 场景 。 类 A 有 两 个 使 用 了 相同 队列 的 方法 (methodA1 和 methodA2 ) 。 
前 一 个 方法 对 某 个 对 象 调用 了 methodd 方法 ， 后 续 会 反 调 回来 。 最 终结 果 还 是 死 锁 。 一 个 
原本 很 有 用 的 方法 dispatch get current queue 已 经 被 弃 用 很 入 了 。 

方案 之 一 是 使 用 dispatch queue set specific 和 dispatch queue get specific 方法 (http:/ 
bit.ly/INOj8fo) ， 但 这 仍然 会 让 代码 变 得 糟糕 。 

要 想 实现 线程 安全 、 不 死 锁 且 易 于 维护 的 代码 ， 强 烈 建议 使 用 异步 风格 。 使 用 Promise 是 
最 好 的 方式 。ReactiveCocoa (参见 4.5 节 ) 为 Objective-C 5| A T FRP 风格 。dispatch_ 


a 




















sync 不 受 这 一 行为 的 影响 。 





ü 





E 11: dispatch get current queue, Developer Tools Manual Page (http://apple.co/ I RPh8SH). 
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PromiseKit 


PromiseKit (https://github.com/mxcl/PromiseKit) 是 支持 使 用 Promise 的 另外 一 个 库 。 
它 甚 至 做 得 更 好 ， 因 为 它 可 以 帮助 避免 代码 向 右 漂移 。 


例 4-15 向 右 漂移 的 Promise 
[CCl 

[[HPNetworkService sharedInstance] promise:rqi] 

subcribeNext:^(id data1) { 
return [[HPNetworkService sharedInstance] promise:rq2]; 

)] 

subscribeNext:^(id data2) { 
return [[HPNetworkService sharedInstance] promise:rq3]; 

H 

subscribeNext:^(id data3) { 
// 此 处 有 三 级 缩 进 
// 查看 开 括号 '[[[[ 














)1 
Hs 


Gil 4-16 不 向 右 漂移 的 Promise 
[NSURLConnection promise:rqi].then(^(id data1){ 
return [NSURLConnection promise:rq2]; 
J).then(^(id data2){ 
return [NSURLConnection promise:rq3]; 
J).then(^(id data3){ 
// 哇 ! 代码 看 起 来 很 连贯 ! 
]); 


注意 ， 在 例 4-15 中 ， 如 果 多 个 Promise 被 链 式 调用 ， 则 会 有 一 系列 的 中 括号 (D. ， 如 
果 你 是 一 位 每 个 中 括号 一 行 且 增 加 一 个 缩 进 的 程序 员 ， 则 代码 会 立刻 向 右 漂移 许多 。 
另 一 方面 ， 在 例 4-16 中 ， 代 码 总 是 在 第 一 列 对 齐 。 


PromiseKit 也 提供 了 优雅 的 错误 处 理 能 力 。 强 烈 建议 深入 研究 PromiseKit, 











4.6 小结 

无 法 想象 哪个 应 用 可 以 不 使 用 并 行 编程 ， 即 使 像 动画 一 样 简单 的 操作 也 需要 多 任务 。 一 切 
耗 时 的 任务 (如 网 络 和 IO) 都 必须 在 后 台 线 程 内 完成 。 

通过 对 现 有 不 同方 案 ( 即 线程 、GCD、 操 作 和 队列 ) 的 深入 分 析 ， 你 现在 应 该 能 够 选择 一 
个 更 加 符合 具体 场景 的 方案 进行 工作 。 

选择 正确 的 方案 来 保证 代码 的 线程 安全 ， 是 实现 应 用 状态 正确 的 关键 。 使 用 信号 量 同步 访 
问 代 码 块 非常 重要 ;， 使 用 读 - 写 锁 实现 高 吞吐 量 的 读 和 有 保护 的 写 同 样 重要 。 

通过 阅读 这 一 部 分 ， 你 已 经 熟悉 了 一 些 核心 优化 技术 ， 如 内 存 管理 、 电 量 使 用 和 并 行 编 
程 。 现 在 你 可 以 对 应 用 的 模型 和 业务 逻辑 层 进行 优化 了 。 























第 三 部 分 
i0S 性 能 





第 二 部 分 为 创建 高 性 能 应 用 英 定 了 基础 。 高 性 能 应 用 就 是 了 解 资源 利用 并 遵循 优化 的 最 佳 
实践 的 应 用 。 虽 然 我 们 的 讨论 围绕 着 iOS 应 用 开发 ， 但 通用 原则 适用 于 任何 Objective-C 
应 用 。 

这 一 部 分 的 章节 将 会 关注 iOS 应 用 开发 中 特有 的 选项 和 技术 。 我 们 将 主要 探讨 以 下 主题 ; 
。 应 用 的 生命 周 其 

* UI 

。 网 络 

。 数据 共享 

。 安全 














第 5 章 


应 用 的 生命 周期 





iOS 应 用 启动 时 会 调用 UIApplicationMain 方法 ， 并 传人 UIApplicationDelegate 类 的 引用 。 
委托 接收 应 用 范围 的 事件 ， 并 且 有 明确 的 生命 周期 ，application:didFinishLaunchingwit 
hoptions: 方法 表明 应 用 已 经 启动 。 关 键 组 件 的 初始 化 就 发 生 在 这 个 方法 中 ， 如 月 江上 报 、 
网 络 、 日 志 以 及 埋 点 的 初始 化 。 此 外 ， 初 次 启动 或 恢复 前 置 状 态 以 便 后 续 启动 时 ， 还 可 能 
会 执行 一 些 一 次 性 的 初始 化 操作 。 


应 用 的 窗口 有 一 个 rootViewControLler， 可 以 驱动 展示 给 用 户 的 UI。 对 应 的 UIViewController 
对 象 同样 具有 明确 的 生命 周期 。 


应 用 启动 过 程 中 的 一 系列 活动 会 影响 初始 加 载 时 间 ， 因 此 ， 为 了 获得 更 好 的 用 户 体验 ， 必 
须 使 初始 化 活动 的 数量 最 小 化 。 但 并 不 能 任性 地 移 除 初始 化 的 任务 ， 因 为 如 果 因 此 而 导致 
应 用 启动 之 后 的 后 续 操 作 变 得 更 慢 ， 肯 定 会 车 恼 用 户 。 


本 章 将 深入 讨论 应 用 的 生命 周期 。 我 们 会 将 应 用 开发 人 员 使 用 事件 回调 的 目的 ， 与 事件 回 
调 的 主要 意图 及 对 应 用 性 能 的 影响 进行 对 比 。 我 们 也 将 回顾 一 些 可 以 取悦 用 户 的 技术 、 秘 
RAT I. 

我 们 将 在 第 6 章 中 探索 UIViewController 的 生命 周期 。 


5.1 应 用 委托 


应 用 委托 通常 E TAT 它 为 应 用 提供 一 些 环 境 变量 ， 其 中 包括 应 用 启动 
的 详细 信息 、 远程 通知 、 REHE , 等 等 。 


如 果 需 要 回顾 应 用 的 结构 和 执行 状态 ， 可 以 参阅 http://apple.co/11V94sL, 
图 5-1 展示 了 在 不 同 状态 之 间 切 换 时 对 应 的 应 用 委托 回调 。 







































































119 













application:shouldRestoreApplicationState: 
application:didDecodeRestorableStateWithCoder 


1 
= application:openURL:source:annotation: 
1 








application:WillResignActive — RK------ i 









application:DidEnterBackground: 


application:shouldSaveApplicationState: 
application:willEncodeRestorableStateWithCoder: 
1 































































































国 应用 未 启动 时 的 用 户 行为 一 一 一 > 正常 控制 流 ， 出 现 频率 较 高 
| 通常 无 需 处 理 的 事件 ------ > 应 用 运行 时 的 openURL 流 程 
上 ”|] 几乎 不 会 被 处 理 的 事件 区 导致 应 用 失去 焦点 的 通知 
国 | 被 处 理 的 事件 ----- 这 附加 流程 ， 由 openURL 引 发 
四 | 推送 至 设备 的 零散 信息 ---- -之 后 台 任务 事件 

E 主 运行 循环 ， 视 图 控制 器 进行 处 理 














5-1: 应 用 委托 回调 的 调用 





虽然 图 5-1 整体 看 起 来 比较 复杂 ， 但 每 个 部 分 都 是 我 们 熟悉 的 内 容 ，application:didDecod 
eRestorableStateWithCoder: 和 application:willEncodeRestorableStateWithCoder: 这 两 个 
方法 可 能 除外 ， 因 为 它们 很 少 被 调用 。 但 是 应 用 通常 都 会 创建 本 地 状态 管理 ， 可 以 从 那里 




















ET. 
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恢复 之 前 的 状态 。 将 这 些 方法 都 加 进来 是 为 了 让 整个 图 看 起 来 更 加 完整 。 





还 有 一 些 其 他 的 事件 回调 并 未 在 图 中 列举 出 来 ， 与 推送 通知 有 关 的 回 




















在 5.3 市 中 探讨 。 











我 们 将 一 次 分 析 一 个 回调 ， 检 查 回调 中 的 代码 ， 并 寻找 是 否 存在 更 好 的 实现 方式 。 








5.2 ”应 用 启动 





调 被 省 略 了 ， 我 们 将 


著名 的 application:didFinishLaunchingWithOptions: 方法 是 应 用 启动 时 最 核心 的 地 方 。 此 
处 不 能 发 生 任 何 错误 ， 且 绝 不 能 发 生 月 涡 ， 否 则 应 用 将 无 法 正常 使 用 ， 直 到 下 次 升级 。 一 
旦 发 生 这 种 情况 ， 如 果 应 用 对 用 户 来 说 并 非 至 关 重要 ， 那 用 户 青 定 就 会 放弃 使 用 这 款 应 用 。 
上 述 方法 会 载 人 所 有 的 依赖 ， 并 初始 化 应 用 的 核心 。 在 启动 时 ， 你 必须 使 该 方法 的 执行 时 





间 尽 可 能 短 ， 








因为 你 不 希望 用 户 等 待 一 段 时 间 UI 才 展 现 H 











HH 来。 同时 你 也 不 希望 应 用 因此 


被 贴 上 “笨拙 ”“ 庞 大 ” “缓慢 ”这 样 的 标签 ， 当 然 也 不 想 在 App Store 里 得 到 糟糕 的 评价 。 


应 用 有 四 种 启动 类 型 。 
。 首次 启动 





安装 应 用 后 的 首次 启动 。 此 时 没有 之 前 的 状态 ， 也 没有 本 地 缓存 。 


这 意味 着 将 会 出 现 以 下 两 种 情况 中 的 一 种 : 没有 需要 加 载 的 内 容 ( 因 
E), 或 者 需要 从 服务 器 上 下 载 初始 数据 (可 
在 应 用 首次 启动 时 ， 你 可 以 选择 提供 引导 








Dropbox 的 引导 图 。 





能 需要 很 长 的 加 载 时 间 )。 








a 








来 总 结 应 用 的 功能 和 用 法 。 








此 加 载 时 间 会 缩 


图 5-2 展示 了 




















5-2; Dropbox 的 引导 图 
。 冷 启动 


应 用 后 续 的 启动 。 在 启动 期 间 ， 可 能 需要 恢复 原来 的 状态 ， 例 如 ， 游 戏 中 达到 的 








ELA 
最 高 等 


级 、 消 息 应 用 中 的 聊天 记录 、 新 闻 应 用 中 上 一 次 同步 的 文章 、 已 登录 用 户 的 证 书 ， 或 者 





仅仅 是 用 户 已 经 使 用 过 的 引导 图 标记 符 。 
图 5-3 展示 了 冷 启 动 时 的 Facebook 应 用 。 注 意 一 下 它 是 如 何在 从 服务 器 获取 更 新 的 同 

















时 快速 载 入 缓存 内 容 的 。 
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(4 Status (Si Photo 











P - S — 
< b m 
hed 18 minutes ago 


7 Likes 2 Comments 


if Like Wi Comment 


F TORO S SS 
6 minutes ago 


yours :) Have a great year ahead 


17 minutes ago 


of joy for all of us. 


E o 





WISH ALL MY FACE BOOK FRIENDS A VERY VERY 
HAPPY NEW YEAR 2015. GOD BLESS YOU ALL 


@ Share 


A very happy and prosperous new year to you and 


1 Like 
i Like Wi Comment @ Share 
EO Doo & celebrating New Year's 
Ld Eve 


Happy New Year to all my friends. 2015 will bring lot 


News Feed Requests Messages Notifications More 





5-3; 冷 启 动 时 的 Facebook 应 用 


热 〈 重 ) 启动 


这 是 指 当 应 用 处 于 后 台 ， 但 并 未 被 挂 起 或 关闭 时 ， 用 户 切 换 至 应 用 而 触发 的 启动 。 在 这 
种 情况 下 ， 当 用 户 通过 点 击 应 用 图 标 或 深层 链接 (参见 8.1 节 ) 返回 应 用 时 ， 不 会 触发 
启动 时 的 回调 ， 而 是 直接 用 applicationDidBecomeActive: (或 application:openURL:so 























urce:annotation:) 回调 。 





通常 来 说 ， 这 种 情况 和 继续 执行 没什么 区 别 ， 只 是 视图 














事件 ， 对 此 ， 我 们 将 在 本 章 后续 部 分 进一步 探讨 。 








控制 器 可 能 需要 处 理 








T 





些 额外 的 








。 升级 后 的 启动 
应 用 升级 以 后 的 启动 。 通 常 而 言 ， 升 级 后 的 启动 与 冷 启 动 没有 差别 。 但 是 ， 不 同 的 启动 
叫 法 表明 了 本 地 存储 发 生变 化 的 时 刻 是 不 同 的 ， 这 些 变化 包括 模式 、 内 容 、 之 前 版 本 挂 
起 的 同步 操作 ， 以 及 内 部 的 API 默认 依赖 。 


5.2.1 首次 启动 

首次 启动 时 ， 应 用 通常 会 执行 多 个 任务 : 

。 加 载 应 用 的 默认 项 (NSUserDefaults、 捆 绑 的 配置 等 ) 

。 检查 私有 /测试 版 本 

。 初始 化 应 用 标识 符 ， 包 括 但 不 限于 对 匿名 用 户 使 用 的 供应 商标 识 符 (Identifier for 
Vendor，IDFV)、 广 告 标识 符 (Identifier for Advertiser, IDFA) 等 

。 MAAC AAR AB 

。 建立 A/B 测试 

。 建立 分 析 方 法 

。 使 用 操作 或 GCD 建立 网 络 

。 建立 Ul 基础 设施 〈 导 航 、 主 题 、 初 始 UI) 

。 显示 登录 提示 或 从 服务 器 加 载 最 新 内 容 及 其 他 更 新 

。 建立 内 存 缓存 (如 图 片 缓存 ) 


上 述 列举 的 内 容 只 是 应 用 在 首次 启动 时 可 能 执行 的 任务 。 其 中 一 些 还 会 在 后 续 启动 中 执 
行 。 问 题 是 ， 任 务 数量 的 快速 增加 必然 会 导致 应 用 的 启动 速度 变 慢 。 

如 果 以 某 一 应 用 为 背景 介绍 这 些 任 务 ， 类 似 的 代码 如 例 5-1 所 示 。 

例 5-1 应 用 的 启动 代码 


-(BOOL)application:(UIApplication *)application 
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { @ 





















































NSString *deviceId = [[[UIDevice currentDevice] 
identifierForVendor] UUIDString]; @ 


NSUserDefaults *defaults - [NSUserDefaults standardUserDefaults]; 
BOOL firstLaunch = ![defaults boolForKey:@"appLaunched"]; © 
if(firstLaunch) ( 

[defaults setBool:YES forKey:Q"appLaunched"]; @ 

[defaults synchronize]; 


// 将 设备 注册 到 服务 器 € 





// 用 设备 ID 建立 A/B 测 试 @ 





[Flurry startSession:Q"API KEY"]; 

[[NSURLCache sharedURLCache] setMemoryCapacity:(8 * 1024 * 1024)]; 
[[NSURLCache sharedURLCache] setDiskCapacity:(50 * 1024 * 1024)]; 
[SDImageCache sharedImageCache].maxCacheSize = 8 * 1024 * 1024; 
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NSString *accessToken = nil; @ 
if(!firstLaunch) { 
accessToken = [defaults stringForKey:Q"accessToken"]; 


} 
if(accessToken) { © 
// 用 户 登 录 





) else { 


if(firstLaunch) ( @ 
// 首 次 启动 
} else { @ 








// 用 











j 


return YES; 


j 


户 未 登录 


-(BOOL)applicationDidBecomeActive:(UIApplication *)application { @ 


#ifdef RELEASE_BETA [3] 
[[BITHockeyManager sharedHockeyManager ] 
configureWithIdentifier:Q"API KEY"]; 
[[BITHockeyManager sharedHockeyManager] startManager]; (D 


#endif 
} 


Q 每 个 应 用 启动 时 都 会 调用 一 次 application:didFinishLaunchingwithOoptions: 回调 。 


@ 获取 IDFV 来 唯一 地 跟踪 此 设 




















O 确定 应 用 是 初次 启动 还 是 之 前 就 已 经 启动 过 了 。 
O 设置 标记 ， 表 明 应 用 之 前 已 经 启动 过 了 。 


Q9 可 以 将 ID 发 送 到 服务 器 进行 匿名 登记 ， 或 用 来 跟踪 唯一 用 户 或 设备 数量 。 


Q 一 些 子 系统 可 能 需要 ID， 例 如， 为 了 进行 A/B 测试 。 

O 其 他 设置 : 分 析 方 法 、 网 络 缓存 、 图 片 缓存 ， 等 等 。 

© 追踪 用 户 登 录 状 态 的 访问 令 牌 。 

O 如 果 访 问 令 牌 存在 ， 则 表明 用 户 已 经 登录 过 。 如 果 用 户 修改 了 密码 或 从 其 他 设备 远程 登 
出 ， 则 需要 刷新 令 牌 。 为 简洁 起 见 ， 此 处 省 略 了 部 分 代码 。 






































© 如 果 访 问 令 牌 不 存在 ， 则 存在 两 种 情况 : 甚 一， 应 用 是 初次 启动 ， 此 时 或 许 需要 展示 应 
用 引导 图 ; 其 二 ， 用 户 在 以 前 的 应 用 会 话 期 间 并 未 登录 。 











D 如 果 用 户 没 有 登录 且 应 用 也 不 是 初次 启动 ， 你 可 以 直接 展示 登录 表单 。 
Q 一 些 配置 项 可 能 要 等 应 用 回 到 前 台 时 (激活 状态 ) 才能 被 设置 。 


需要 注意 的 是 ， 每 次 应 用 











从 后 台 切 换 至 前 台 时 ， 此 方法 都 会 被 调用 ， 并 非 只 是 启动 时 。 





因此 ， 你 要 确保 不 运行 动画 以 免 引 入 重复 延迟 ， 因 为 这 可 能 会 车 恼 用 户 。 
(9 RELEASE BETA 不 是 标准 的 标记 ， 而 是 自 定义 标记 ， 用 来 区 分 是 App Store 版 本 还 是 私有 
发 布 。 为 了 保证 正常 运行 ， 你 需要 创建 多 个 配置 项 / 目标 ， 如 
@ 在 本 例 中 ，HockeyKit (https//www.hockeyapp.net/) 设置 已 经 完成 ， 应 用 可 以 执行 其 他 





任务 了 。 





图 5-4 所 示 。 




















Y Configurations 


Name 
> Debug 
b Release 
> Distribution 
> beta 


Based on Configuration File 
2 Configurations Set 

1 Configuration Set 

1 Configuration Set 

1 Configuration Set 





+ 


Use Release for command-line 
Y Code Signing Identity 
Debug 
Any iOS SDK ¢ 
Distribution 
Any iOS SDK ¢ 
Release 
Any iOS SDK ¢ 
beta 
Any iOS SDK ¢ 


Y Apple LLVM 6.0 - Preprocessing 


Setting B Aerogram 
Enable Foundation Assertions Yes $ 
Enable Strict Checking of objc msgSend Calls No? 
Y Preprocessor Macros 
Debug DEBUG-1 
Distribution 
Release o 
beta RELEASE BETA-1 


builds 


«Multiple values» 人 
Don't Code Sign $ 
iOS Developer > 
iOS Distribution 7 
iOS Distribution > 
Don't Code Sign $ 
iOS Developer > 
iOS Distribution > 
iOS Distribution > 


Preprocessor Macros Not Used In Precompiled Headers 








5-4, 多 种 配置 
注意 ， 从 各 个 方面 来 看 ， 这 个 列表 都 还 不 够 完整 。 


广告 相关 的 初始 化 、 日 志 、 














应 用 安装 的 


属性 、 单 点 登录 等 其 他 任务 都 没有 展示 出 来 。 其 实 这 很 大 程度 上 取决 于 应 用 的 需求 和 结构 。 











虽然 每 个 子 系统 都 有 较 高 的 性 能 ， 但 一 起 使 用 时 ， 整 体 的 性 能 可 能 会 下 降 。 
* 例如 ， 如 果 多 个 组 件 试图 同时 从 文件 系统 读 取 ， 则 一 定 会 导致 整体 变 得 迟缓 。 








这 样 就 会 出 现 之 前 提 到 的 情景 一 一 子 系 统 的 初始 化 需要 大 量 的 时 间 ， 而 它们 彼此 之 间 可 能 又 
存在 依赖 。 例 如 ， 主 题 可 能 依赖 A/B 测试 ， 数 据 同步 可 能 依赖 令 牌 是 否 有 效 (又 名 登录 )。 


如 果 这 些 (可 能 是 强制 的 ) 初始 化 加 起 来 要 耗费 很 长 时 间 ， 我 们 该 如 何 优化 呢 ? 











Ab 





， 再 从 此 处 着 手 。 











你 可 以 遵循 下 述 具 体 步 又 ， 拆 解 任务 列表 ， 从 而 获得 更 高 的 性 能 。 


个 问题 没有 明确 的 答案 。 其 中 一 个 方法 是 退 一 步 ， 先 确定 要 展示 应 用 UI 所 需 的 最 低 要 
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(1) 确定 在 展示 UI 前 必须 执行 的 任务 。 
如 果 应 用 是 第 一 次 启动 ， 那 么 没有 必要 加 载 任何 用 户 偏好 ， 如 主题 、 刷 新 间隔 、 缓 存 大 
小 等 。 此 时 是 没有 任何 自 定义 值 的 。 初 始 缓存 肆意 增长 也 是 没 问 题 的 ， 因 为 它 的 增长 不 
会 超过 最 终 的 限制 值 。 
月 潢 报告 系统 应 该 第 一 个 被 初始 化 。 
(2) 按 顺序 执行 任务 。 
排序 是 非常 重要 的 ， 因 为 任务 之 间 可 能 具有 相互 依赖 性 ， 同 时 ， 排 序 还 可 以 节省 用 户 的 
宝贵 时 间 。 
例如 ， 如 有 果 先 触发 了 访问 令 牌 的 验证 操作 ， 那 么 其 他 任务 可 能 会 并 行 执行 ， 因 为 验证 过 
程 需要 进行 网 络 连接 。 但 是 这 样 就 会 导致 一 种 情况 : 如 果 其 他 任务 先 完成 ， 而 验证 还 未 
完成 ， 应 用 就 必须 等 待 验证 完成 才能 继续 执行 。 
(3) 将 任务 拆 分 为 两 类 : 一 类 是 必须 在 主线 程 中 执行 的 任务 ， 另 一 类 是 可 以 在 其 他 线程 中 执 
行 的 任务 ， 然 后 分 别 执行 。 
还 可 以 进一步 将 在 非 主 线程 中 执行 的 任务 分 为 可 以 并 发 执行 的 和 不 能 并 发 执行 的 。 
(4) 其 他 任务 可 以 在 加 载 UI 后 执行 或 异步 执行 。 
延迟 其 他 子 系统 (如 记录 仪 和 分 析 方 法 ) 的 初始 化 。 在 应 用 的 后 续 阶 段 将 一 些 操作 〈 例 
如 ， 写 日 志 消 息 或 跟踪 事件 ) 放 入 队列 中 ， 直 到 子 系 统 完 全 完成 初始 化 。 
你 也 许 会 注意 到 ， 其 实 并 没有 固定 的 解决 方案 。 这 些 优 化 很 大 程度 上 依赖 于 对 子 系统 的 控 
制 方式 。 针 对 崩溃 上 报 、A/B 测试 、 埋 点 和 分 析 、 网 络 、 图 片 缓存 等 子 系统 ， 有 很 多 第 三 
方 解 决 方案 。 优 化 加 载 时 间 的 最 佳 方式 取决 于 选择 的 方法 以 及 对 它们 有 多 少 空间 。 


对 于 应 用 的 依赖 项 ， 如 果 你 能 获取 到 代码 ， 并 且 知 道 修复 方法 ， 那 么 请 提交 
对 应 的 补丁 。 通 过 对 社区 作 贡 献 ， 你 将 帮助 那些 面临 类 似 问 题 的 人 。 
如 果 你 购买 了 许可 证 ， 那 么 要 积极 联系 该 公司 ， 你 应 该 能 得 到 答复 。 如 果 疫 
有 回应 ， 那 就 毫 不 犹豫 地 抛弃 它 吧 ， 寻 找 其 他 可 替代 的 方法 。 最 后 要 明确 的 
是 ， 是 你 的 应 用 在 与 用 户 交互 ， 第 三 方 SDK 在 用 户 眼中 并 不 重要 。 













































































举 个 例子 ， 如 果 你 发 现 分 析 SDK 需要 收集 大 量 数据 (如 操作 系统 版 本 、 应 用 版 本 、 设 备 
信息 等 ) 或 需要 从 本 地 缓存 加 载 一 些 配置 ， 那 么 最 好 在 非 主线 程 异 步 进行 初始 化 ， 同 时 将 
所 有 事件 放 入 一 个 队列 中 (ff NSMutablearray 一 样 简单 ) ， 在 一 次 初始 化 中 全 部 执行 。 


如 果 能 获取 到 代码 ， 打 补丁 就 较为 容易 了 。 如 果 获 取 不 到 ， 你 就 需要 维护 自己 的 队列 。 这 
种 做 法 的 唯一 缺点 是 ， 有 些 事件 可 能 会 出 现 不 正确 的 时 间 戳 和 位 置 。 在 某 些 情况 下 ， 时 间 
惟有 几 毫 秒 的 偏差 或 定位 有 几米 的 偏差 都 是 可 接受 的 。 但 也 有 一 些 对 准确 度 要 求 极 高 的 情 
况 〈 如 果 不 准确 ，SDK 会 表现 很 差 ) ， 那 可 能 就 需要 另 做 选择 了 。 

例 5-2 中 的 代码 提供 了 一 种 可 以 最 大 程度 减少 加 载 时 间 的 方法 。 该 示例 列 出 了 当代 码 不 可 
用 时 ， 异 步 初始 化 分 析 SDK 的 必要 步骤 。 

有 些 SDK 需要 在 主线 程 中 进行 初始 化 。 多 留意 这 些 情况 ， 因 为 这 会 直接 影响 应 用 的 加 载 
时 间 。 
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例 5-2 ”应 用 加 载 时 间 的 优化 


//HPInstrumentation.m 
@interface HPInstrumentation () @ 


@property (nonatomic, copy) BOOL initialized; 
@property (nonatomic, strong) NSMutableArray *events; 
(property (nonatomic, strong) dispatch queue t queue; 


-(void)markInitialized; 
*(void)logEventImpl:(NSString *)name; 


Qend 
static HPInstrumentation *_instance; @ 
(implementation HPInstrumentation 


*(HPInstrumentation *)sharedInstance { © 
return instance; 


} 


*(void)setSharedInstance:(HPInstrumentation *)instance { @ 
_instance = instance; 


} 


*(void)logEvent:(NSString *)name { @ 
[[HPInstrumentation sharedInstance] logEventImpl:name]; 


} 


-(instancetype)initWithAPIKey:(NSString *)apiKey { @ 
if(self = [super init]) { 
self.initialized = NO; 
self.events = [NSMutableArray array]; 
self.queue = dispatch_queue_create("com.m10v.queue.analytics", 
DISPATCH QUEUE CONCURRENT); 


dispatch async( @ 
dispatch get global queue(DISPATCH QUEUE PRIORITY DEFAULT, 0), “{ 
[Flurry startSession:apiKey]; 
dispatch sync barrier(self.queue, ^( © 
for(NSDictionary *name in self.events) { 
[Flurry logEvent:name]; 
} 
self.events = nil; © 
self.initialized = YES; 
H; 
95 
} 
return self; 


} 


-(void)logEventImpl:(NSString *)name ( @ 
dispatch sync(self.queue, “{ @ 





应 用 的 生命 周期 | 127 


if(self.initialized) ( @ 
[Flurry logEvent:name withParameters:params]; 


} else { 
[self.events addObject:name]; 


p; 
} 


Qend 


//HPAppDelegate.m 
-(BOOL) application: (UIApplication *)application 
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { 
HPInstrumentation *analytics - [[HPInstrumentation alloc] 
initWithAPIKey:Q"API KEY"]; (9 


[HPInstrumentation setSharedInstance:analytics]; (D 
[HPInstrumentation logEvent:Q"App Launched"]; (9 
} 


@ HPInstrumentation 是 对 底层 埋 点 API 的 包装 。 它 一 直 将 事件 保存 于 内 存 之 中 ， 直 到 底 

E SDK (此 处 指 的 是 Flurry) 被 初始 化 。 

6 因为 在 底层 SDK 准备 好 之 前 ， 我 们 需要 在 初始 化 时 做 一 些 调整 ， 所 以 伪 单 例 模式 是 首 
选 。_instance 是 可 以 被 设置 或 重 置 的 单 例 。 

© 共享 实例 / 单 例 的 获取 方法 。 这 是 一 个 公共 方法 (在 .h 文件 中 声明 )。 

O 共享 实例 / 单 例 的 设置 方法 。 这 也 是 一 个 公共 方法 。 
需要 注意 的 是 ， 如 果 任 何 代码 都 可 以 使 用 设置 方法 ， 那 么 可 能 会 导致 滥用 。 因 此 ， 必 须 

谨慎 使 用 。 

O 设置 日 志 事 件 的 方法 是 公共 的 类 方法 ， 不 会 发 生变 化 ， 而 会 使 更 新 向 后 兼容 。 
至 于 实现 ， 它 采用 了 非 公共 的 实例 方法 VogEventImpl, 

@@ 类 的 自 定 义 初 始 化 器 。 

@ 除 了 初始 化 状态 ， 它 调用 dispatch_async 来 初始 化 底层 SDK”( 此 处 指 的 是 Flurry 
SDK), 

© — HIKE SDK 被 初始 化 ， 则 刷新 所 有 排队 的 events, 使 用 queue 获取 写 入 锁 ， 从 而 确 
保 当 事件 列表 被 刷新 时 ， 甚 他 事件 不 被 添加 至 列表 中 。 

O 释放 内 存 。 

@ logEventImpl 方法 的 实现 。 

@ 使 用 步骤 8 中 同样 的 queue， 获 取 一 个 读 取 锁 以 保证 并 发 写 入 。 

Q 如 果 SDK 尚未 被 初始 化 ， ET 否则 直接 将 日 志 写 到 底层 SDK. 

@ HPInstrumentation 在 应 用 委托 中 被 实例 化 一 
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TE 1: 单 例 模 式 是 指 实体 的 一 个 实例 ， 有 多 种 实现 方式 。 正 如 我 们 在 第 2 章 中 讨论 的 ， 应 尽量 避免 应 用 范围 
内 不 可 复位 的 单 例 。 

注 2: f 非 主线 程 中 初始 化 。 

注 3: 另 一 个 隐 含 假设 -一 生成 新 事件 的 速度 比 刷新 事件 的 速度 要 慢 得 多 。 如 果 不 是 ， 你 有 可 能 错误 地 使 用 了 
分 析 。 再 考虑 考虑 。 
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Q 设置 共享 实例 。 
Q 调用 logEvent 方法 写 日 志 。 此 处 不 需要 任何 改动 。 


注意 ， 例 5-2 中 的 实现 仅仅 是 多 种 备 选 方案 之 一 。 


使 用 委托 进行 初始 化 是 一 种 更 好 的 方式 。 初 始 委托 添加 至 列表 中 ， 其 他 则 使 用 实际 的 
SDK. fil 5-3 提供 了 可 用 于 初始 化 和 切换 的 示例 代码 。 


实现 方式 遵循 状态 设计 模式 。 这 种 方法 的 优点 在 于 可 以 很 容易 地 管理 底层 实现 。 
例 5-3 使 用 委托 进行 初始 化 


-(instancetype)initWithAPIKey:(NSString *)apiKey { 
//…… 用 于 设置 的 相同 代码 




















self.delegate = [[HPInstrumentationUselist alloc] init]; Q 


// 以 下 代码 用 于 初始 化 后 
dispatch sync barrier(self.queue, “{ 
for(NSDictionary *name in self.delegate.events) { @ 
[Flurry logEvent:name]; 





self.delegate = [[HPInstrumentationUseSDK alloc] init]; © 
35 
} 


-(void)logEventImpl:(NSString *)name { 
dispatch sync(self.queue, “{ 
[self.delegate logEvent:name]; @ 
DE 
} 


@ 首先 ， 委 托 指向 一 个 对 象 ， 该 对 象 将 事件 排 和 列表 。 

O 一 旦 准备 完毕 ， 刷 新 已 经 排列 好 的 事件 …… 

©- 同时 改变 委托 的 指向 ， 将 其 指向 使 用 分 析 SDK 的 对 象 。 

@ logEventImpl 更 加 简单 ， 它 使 用 委托 记录 日志。 不 再 需要 根据 当前 的 状态 (是 否 初始 化 
成 功 ) 作出 决定 。 


5.2.2 Wa 

上 述 的 介绍 应 该 让 你 对 应 用 启动 时 执行 的 任务 有 了 初步 的 了 解 。 冷 启动 期 间 执行 的 任务 只 
有 很 小 的 变化 ， 却 有 可 能 产生 很 大 的 影响 。 

冷 启动 中 一 个 较为 重要 的 任务 是 ， 裁 入 之 前 的 状态 。 在 应 用 中 ， 显 示 给 用 户 (登录 后 ) 的 
第 一 个 画面 是 feed 流 。 如 果 用 户 在 以 前 的 启动 中 登录 过 ， 并 且 数 据 已 经 同步 ， 那 我 们 就 会 
考虑 加 载 之 前 已 经 缓存 的 feed 流 。 

我 们 将 会 在 第 7 章 和 第 8 章 深入 讨论 本 地 缓存 的 相关 方法 。 此 处 先 假设 我 们 使 用 了 一 些 方 
法 ， 可 以 在 记录 表 里 做 一 些 基 本 的 CRUD 操作 。 本 市 主要 讨论 如 何 较 好 地 利用 那些 方法 。 
为 了 实现 向 用 户 展示 feed 流 的 任务 ， 必 须 向 服务 器 请 求 最 近 的 更 新 ， 同 时 还 要 从 本 地 缓存 
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加 载 数 据 。 这 些 行 为 是 不 用 思考 就 知道 的 。 但 是 ， 以 下 几 点 却 是 不 容 忽视 的 。 


。 展示 有 用 且 有 意义 的 UI 所 需要 的 最 少 信息 数目 (min). 

。 记录 从 本 地 缓存 加 载 M 条 信息 花费 的 时 间 〈 记 作 d). 

。 记录 从 服务 器 获取 最 新 的 M 条 信息 花费 的 时 间 〈 记 作 ir), 

。 为 了 获得 更 快 的 速度 , 任何 时 刻 在 内 存 中 存储 的 最 大 信息 数目 (max) ， 特 别 是 在 快速 请 
SARI 











如 果 不 能 在 3 秒 内 加 载 M 条 信息 ， 那 么 用户 体验 将 显著 下 隆 。* 








这 些 值 有 助 于 为 应 用 启动 时 的 数据 检索 定义 一 个 具体 的 策略 。 
考虑 以 下 儿 种 情况 : 


(1)#1=3 秒 , tr=1 秒 
(9)WU=1.5 秒 ,tr=1.5 秒 
(3)#=1 秒 ,tr=3 秒 


假设 min=5，max=20。 我 们 将 针对 这 些 时 间 对 不 同 的 M 值 进行 讨论 。 
在 我 们 的 案例 研究 中 ， 视 图 层级 如 图 5-5 所 示 ， 我 们 将 按照 此 种 情况 进行 时 间 测 量 。 
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图 5-5: 视图 控制 器 结构 


注 4: J. O Dell, VentureBeat, "This Is Why Users Think Your Mobile App Sucks: A 3-Second Response Time" (http:// 
venturebeat.com/2013/09/02/this-is-why-users-think-your-mobile-app-sucks-a-3-second-response-time/). 
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注意 ， 实 际 的 容器 层 是 window。 根 视图 控制 器 HPMainTabBarViewController 包含 了 三 个 子 控 
il] 器 : HPChapterViewController, HPDebugLogViewController 和 HPSettingsViewController, 
图 5-5 只 展示 出 了 当前 可 见 的 视图 控制 右 。 子 控制 器 也 存在 ， 只 是 不 可 见 而 已 。 

1. 情景 1 

如 果 从 远程 服务 器 同步 数据 比 从 本 地 缓存 读 取 更 快 ， 那 么 你 应 该 尽早 地 触发 同步 任务 。 在 
Cocoa 推 党 的 典型 MVC 架构 中 ， 对 应 的 UIViewController 负责 触发 任务 ， 因 为 视图 控制 
器 知道 需要 何 种 数据 来 填充 UI。 一 种 可 能 的 实现 方式 是 ， 在 应 用 的 委托 中 创建 服务 ， 将 
服务 注入 视图 控制 器 ， 然 后 由 视图 控制 器 触发 服务 。 当 然 ， 如 果 想 要 市 省 额外 的 儿 毫 秒 ， 
从 应 用 委托 触发 同步 也 是 没有 问题 的 。 创 建 视图 控制 器 或 只 加 载 shell UI 都 需要 花费 一 
些 时 间 。 

如 果 以 ReactiveCocoa 框架 为 基础 ,真正 实现 起 来 应 该 不 会 太 复杂 。 我 们 可 以 创建 一 个 信 
号 ， 将 其 传递 给 视图 控制 器 ， 在 需要 时 向 下 传递 。 一 旦 数据 可 用 ， 视 图 控制 器 就 会 收 到 通 
知 。 此 外 ， 除 了 网 络 不 可 用 的 特殊 情况 ， 其 他 时 候 都 可 以 正常 地 使 用 本 地 缓存 。 

在 第 一 个 分 情景 中 ， 假 设 检 索 大 量 信 息 所 需 的 时 间 能 够 完全 达到 甚至 超出 向 用 户 提供 较 好 
用 户 体验 这 一 目标 所 需 的 最 小 值 。 这 意味 着 ， 一 旦 服务 器 数据 可 用 ， 那 么 就 无 需 其 他 操 
作 。 因 为 数据 可 以 直接 提供 给 应 用 并 触发 刷新 。 


例 5-4 展示 了 如 何 使 用 Promise, Promise 开始 于 应 用 委托 ， 但 在 视图 控制 器 中 使 用 。 这 种 
做 法 似乎 违反 了 MVC 的 原则 ， 所 以 很 多 人 不 赞同 。 换 一 种 角度 看 ， 这 其 实 只 是 一 个 依赖 
TEA. Promise 是 注入 视图 控制 器 的 数据 源 。 


例 5-4 使 用 数据 源 注入 的 视图 控制 器 
//HPAppDelegate.m 
-(BOOL) application: (UIApplication *)application 
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions { 




















































































































// 验 证 访问 令 牌 .用 户 登录 等 
RACSignal *feedSignal = [[[HPSyncService sharedInstance] 
fetchType:Q"feed"]] replay]; @ 


HPUserFeedViewController *viewController - 
(HPUserFeedViewController *) self.window.rootViewController; 


viewController.feedSignal = feedSignal; @ 
} 


//HPUserFeedViewController.m 
-(void)viewDidLoad { 
(weakfily(self); 
[[self.feedSignal © 
deliverOn:[RACScheduler mainThreadScheduler]] €) 
subscribeNext:^(HPUserFeed *feed) { 
@strongify(self); 
[self updateWithFeed:feed]; @ 
self.feedSignal = nil; © 
} error:^(NSError *) { 


// 处 理 错误 
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Fl; 
} 


@ 创建 信号 。 

@ 假设 rootViewController 是 HPUserFeedViewController, ， 包 含 一 个 feedSignal 属性 ， 并 

将 其 设置 为 先前 创建 的 信号 。 

© 信号 的 订阅 在 控制 器 中 执行 。 因 为 此 操作 是 在 viewDidLoad 中 完成 的 ， 所 以 可 以 保证 只 

被 执行 一 次 。 

O 响应 的 辅助 调用 在 主线 程 中 运行 。 

© 触发 UI 更 新 。 

© 释放 feedSignaL。 这 步 操作 会 使 引用 计数 减 1， 最 终 会 让 它 析 构 。 

在 其 他 分 场景 中 ， 如 果 得 到 的 信息 数 不 足 以 提供 良好 的 用 户 体验 ， 那 么 你 可 以 选择 以 下 方 

法 之 一 。 

。 提高 服务 性 能 以 获取 足够 的 信息 。 

。 如 果 因 为 某 种 原因 ， 服 务 不 受 控制 ， 那 么 你 需要 寻找 其 他 的 解决 方法 。 看 看 同样 的 内 容 
是 否 能 够 以 不 同 的 方式 重新 泻 染 ， 如 果 新 的 泻 染 结果 能 占用 更 多 的 展示 空间 ， 那 么 就 可 
以 减少 拉 取 的 信息 条 数 ， 也 就 不 需要 通过 增加 空白 空间 来 填充 界面 了 。 
例如 ， 一 张 图 片 本 来 展示 为 小 缩 略 图 ， 但 我 们 可 以 尝试 将 其 放大 ， 这 样 就 能 得 到 一 张 较 
好 的 概要 图 了 。 请 牢记 ， 这 些 新 的 设计 实际 上 可 以 带 来 更 好 的 用 户 体验 。 图 5-6 展示 了 
Facebook 和 Yahoo Finance 的 屏幕 截图 。 通 过 将 左 侧 较 旧 的 截图 与 右 侧 较 新 版 本 的 截 医 
进行 比较 ， 我 们 可 以 看 出 ， 在 较 新 的 截图 中 ， 单 个 条 目 不 仅 占用 了 更 多 的 空间 (信息 密 
度 低 ) ， 而 且 在 视觉 上 也 更 吸引 人 。 

2. 情景 2 

如 果 从 本 地 缓存 加 载 的 时 间 与 从 服务 器 中 检索 所 需 的 时 间 相 当 ， 那 么 最 好 同时 触发 两 者 的 

操作 。 

举 个 例子 ， 一 个 邮件 应 用 可 能 需要 一 段 时 间 才 能 打开 ， 因 为 除了 从 本 地 缓存 加 载 数 据 ， 它 

还 必须 从 服务 器 同步 数据 。 它 从 本 地 加 载 数据 的 时 间 很 可 能 与 获取 新 邮件 的 时 间 相 当 。 

如 之 前 所 述 ， 拉 取 应 该 尽早 地 触发 。 将 信号 注入 视图 控制 器 ， 并 从 那里 获取 数值 。 视 图 控 

制 器 需要 同步 更 新 。 如 果 从 服务 器 端 歼 取 的 数据 是 可 用 的 ， 那 么 本 地 缓存 的 数据 应 该 丢 

弃 ， 毕 况 服 务 器 端的 数据 更 新 一 些 。 


有 了 这 些 改变 ， 视 图 控制 器 的 代码 与 例 5-5 中 的 代码 类 似 。 
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例 5-5 有 多 种 数据 源 注入 的 视图 控制 如 


@interface HPUserFeedViewController () 

@property (nonatomic, copy) BOOL updatedFromServer; @ 
Qend 

(implementation HPUserFeedViewController 


-(void)viewDidLoad { 
self.updatedFromServer - NO; 


(Qweakify(self); 
[[self.cacheFeedSignal @ 
deliverOn: [RACScheduler mainThreadScheduler] 
subscribeNext:^(HPUserFeed *feed) { 
@strongify(self); 
[self updateWithFeed:feed fromServer:NO]; 
self.cacheFeedSignal - nil; 
) error:^(NSError *error) { 
// 处 理 错误 
}]; 





[[self.serverFeedSignal © 
deliverOn: [RACScheduler mainThreadScheduler ] 
subscribeNext:^(HPUserFeed *feed) { 
@strongify(self); 
[self updateWithFeed:feed fromServer:YES]; 
self.cacheFeedSignal - nil; 
} error:^(NSError *error) { 
// 处 理 错误 
}]; 





} 


-(void)updateWithFeed: (HPUserFeed *)feed 
fromServer:(BOOL)fromServer ( @ 


if(self.updatedFromServer) { @ 


return; 
} 
// 刷 新 UI 
self.updatedFromServer = fromServer; (9 
} 
@end 


@ updatedFromserver 是 一 个 私有 属性 ， 用 来 追踪 从 远程 服务 器 获取 的 数据 是 否 引 起 了 更 新 。 

@ 订阅 cacheFeedsignal， 它 可 以 接收 来 自 本 地 缓存 的 数据 。 

© 订阅 serverFeedsignal， 它 可 以 接收 来 自 远 程 服务 器 的 数据 。 

O 更 新 UI 的 方法 新 增 了 一 个 额外 的 参数 一 一 区 分 数据 源 是 否 为 远程 服务 器 的 标记 。 

Q WR UI 已 经 从 服务 器 接收 到 了 更 新 数据 ， 那 就 没有 必要 再 进一步 地 更 新 了 。 当 服务 器 
的 响应 比 从 本 地 缓存 加 载 更 迅速 时 ， 就 会 出 现 这 种 情况 。 
































AR 
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Q 一 旦 完成 刷新 ， 设 置 updatedFromServer 标记 ， 表 明 刷 新 是 否 使 用 了 服务 器 的 数据 。 

如 果 从 某 一 单独 的 数据 源 获得 的 数据 不 足以 提供 良好 的 用 户 体验 ， 最 好 的 做 法 是 结合 两 个 
数据 产 的 结果 ， 然 后 展示 最 终 的 结果 。 总 之 ， 检 索 结 果 所 用 的 时 间 是 相当 的 ， 结 果 的 合并 
对 用 户 而 言 也 是 透明 的 。 

3. 情景 3 

最 常见 的 情境 是 ， 从 本 地 缓存 获取 数据 所 需 的 时 间 比 从 服务 器 上 检索 最 新 数据 花费 的 时 间 
更 短 。 在 这 种 情况 下 ， 应 该 先 加 载 旧 数据 ， 最 新 的 数据 可 用 时 再 更 新 。 

从 实现 的 角度 来 看 ， 解 决 方案 类 似 于 我 们 在 情景 2 中 所 看 到 的 一 一 将 两 个 数据 源 注 入 视 
图 控制 器 中 。 唯 一 的 区 别 是 ， 在 updateWithFeed:fromServer: 方法 中 ， 条 件 语 句 if(self. 
updatedFronServer) 被 执行 的 概率 几乎 为 零 。 


5.2.3” 热 启动 

热 启 动 是 指 切换 到 一 个 已 经 运行 了 的 应 用 。 两 个 原因 可 能 会 使 应 用 变 成 非 激 活 状 态 : 一 是 
用 户 向 下 拉 搜 状态 栏 ， 二 是 用 户 点 击 home 键 或 切换 至 其 他 应 用 。 

热 局 动 有 两 种 情境 : 

。 用 户 点 击 图 标 

。 应 用 接收 到 深层 链接 

1. 应 用 重启 

当 用 户 点 击 应 用 图 标 时 ， 一 般 不 需要 执行 其 他 特殊 的 操作 。 

应 用 处 于 安全 状态 ， 或 者 运行 很 多 动画 时 ， 可 以 监测 背景 和 前 景 通知 。 在 第 一 种 情况 下 ， 
应 用 每 次 进入 前 景 状态 时 ， 都 会 展示 登录 界面 ， 在 后 一 种 情况 下 ， 动 画 或 者 游戏 状态 会 被 
和 暂停， 需要 恢复 。 图 5-7 展示 了 Temple Run 和 Intuit Mint 处 理 这 种 情景 的 方式 。 

此 外 ， 其 他 操作 类 似 于 用 户 和 应 用 继续 进行 交互 时 的 操作 。 

2. 深层 链接 

当 应 用 接收 到 application:openURL:sourceApplication:annotation: 回调 时 ， 期 望 能 跳 转 
到 应 用 的 特定 页 面 ， 实 现 用 户 想 要 完成 的 操作 。 但 此 时 的 目标 应 用 可 能 已 经 发 生变 化 ， 处 
于 某 一 特定 状态 了 。 

如 果 深 层 链 接 需 要 从 服务 器 获取 数据 ， 那 么 可 以 先 展示 与 深层 链接 相关 的 原始 页 面 ， 或 者 
先 展 示 一 个 进度 条 ， 等 从 服务 器 获取 到 了 最 新 数据 ， 再 执行 刷新 操作 。 

为 了 实现 用 户 想 要 完成 的 操作 ， 你 可 以 遵循 下 列 最 佳 实践 。 



































































































































应 用 的 生命 周期 | 135 





1 9 $ 73% G+ 


^mint 


Enter Passcode 


RESUME 














5-7; 应 用 热 重启 一 Temple Run 和 Mint (Temple Run 暂停 了 游戏 并 期 望 用 户 继续 ，Mint 用 密 
码 保证 访问 ) 


。 提供 一 个 “返回 ” 源 应 用 的 选项 ， 通 过 该 选项 支持 深层 链接 。 最 简单 的 方式 是 ， 当 应 用 
处 理 完 相关 事件 后 ， 在 传 入 的 URL 上 增加 一 个 参数 ， 然 后 将 其 当 作 最 终 的 目的 URL. 
举 个 例子 ，Facebook 应 用 可 以 深层 链接 到 Messenger 应 用 。 在 链接 时 提供 一 个 “后 退 ” 
选项 ,一 旦 完成 消息 的 相关 操作 ， 就 可 以 直接 回 退 至 Facebook 应 用 。 

。 实现 应 用 时 ， 最 好 能 将 其 当 作 一 个 简化 的 有 限 状态 机 。 这 样 就 可 以 推送 新 的 屏幕 ， 并 在 
交互 完成 时 直接 弹出 。 

如 上 述 例 子 ， 如 果 打 开 Messenger 应 用 ， 则 其 中 没有 返回 到 Facebook 应 用 的 “返回 ” 
按钮 。 此 时 ， 如 果 深 层 链接 到 Messenger 应 用 ， 就 像 是 直接 “推送 ”到 应 用 。 

当然 ， 这 里 的 目的 是 “与 朋友 聊天 ”。 一 旦 目的 实现 ， 如 果 能 直接 返回 到 应 用 启动 的 地 
方 ， 那 么 会 显得 更 加 友好 一 些 。 需 要 注意 的 是 ， 在 这 种 形式 的 通信 中 ， 两 个 应 用 都 必须 
支持 深层 链接 。 

iPhone 手机 现在 仍然 没有 硬件 后 退 按钮 ， 这 意味 着 应 用 必须 适应 UI 本 身 。 图 5-8 显示 了 
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图 5-8: 有 “返回 ”导航 支持 的 深层 链接 


5.2.4 升级 后 启动 

应 用 升级 后 的 首次 启动 将 遵循 下 列 情形 之 一 : 

。 无 本 地 缓存 或 应 用 完全 放弃 缓存 ; 

。 本 地 缓存 可 用 ， 可 以 直接 使 用 或 需要 切换 至 升级 版 本 。 

如 果 无 本 地 缓存 或 应 用 决定 放弃 缓存 (例如 ， 数 据 不 可 用 或 从 服务 器 同步 获取 更 快 )， 则 

不 需要 进行 特殊 处 理 。 

本 地 数据 发 生 改 变 时 通知 用 户 。 以 下 的 最 佳 实践 可 以 让 用 户 有 更 好 的 体验 。 

。 如 果 本 地 缓存 可 用 ， 通 知 用 户 该 情况 。 如 果 没 有 迁移 到 本 地 缓存 的 必要 ， 则 无 需 通知 用 
户 ， 因 为 本 地 缓存 的 使 用 是 隐 式 的 。 

。 如 果 必 须 花 几 分 钟 对 数据 进行 迁移 ， 那 么 向 用 户 展示 一 个 可 以 推迟 该 操作 的 选项 。 

。 如 果 从 服务 器 检索 数据 更 快 、 更 容易 ， 因 而 必须 放弃 本 地 缓存 的 使 用 ， 那 么 这 种 情况 下 

需要 通知 用 户 。 
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有 个 现成 的 邮件 应 用 的 例子 ， 将 旧版 本 的 记录 迁移 至 新 版 本 的 升级 模式 时 所 做 的 工作 ， 
要 比 从 服务 器 上 直接 检索 数据 更 加 复杂 。 


5.3 推送 通知 

通知 是 内 容 驱 动 型 应 用 的 一 个 组 成 部 分 。 这 里 的 内 容 可 以 像 新 闻 一 样 组 合 ， 也 可 以 像 邮件 
一 样 由 用 户 生 成 。 实 现 这 些 的 前 提 条 件 是 ， 你 了 解 应 用 委托 中 的 application:didReceiveR 
emoteNotification: 和 application:didReceiveLocalNotification: 回调 。 


同时 ， 你 还 需要 了 解 调 用 方法 的 顺序 ， 以 及 用 户 交 互 是 如 何 驱动 其 他 方法 实现 调用 的 。 这 
两 方面 非常 重要 ， 因 为 它们 会 影响 应 用 初始 化 时 遵循 的 顺序 ， 以 及 特定 回调 中 执行 的 子 组 
件 初 始 化 的 种 类 。 在 特定 回调 中 执行 不 同 子 组 件 的 初始 化 ， 可 以 实现 资源 利用 的 最 小 化 ， 
最 大 化 要 完成 的 任务 的 数量 。 


5.3.1 远程 通知 
图 5-9 显示 了 收 到 通知 时 的 委托 回调 。 















































application:didFinishLaunchingWithOptions: 











图 5-9. 应 用 委托 中 关于 通知 的 完整 生命 周期 
以 下 是 对 iOS 8 生命 周期 的 描述 。 
。 如 果 应 用 是 激活 状态 的 ， 则 通过 didReceiveRemoteNotification 回调 接收 通知 。 
没有 调用 其 他 回调 ， 为 避免 分 心 ， 也 没有 向 用 户 展示 UI。 
。 如 果 应 用 在 后 台 运 行 或 停止 ， 只 有 静默 推送 通知 回调 会 被 触发 。 
基于 通知 设置 ， 非 静默 推送 通知 可 能 会 出 现在 通知 中 心 或 作为 报警 阐 窗 ， 或 更 新 应 用 图 
标的 角 标 计数 。 
。 当 用 户 使 用 通知 中 心 或 报警 弹 窗 开启 通知 时 ， 可 能 会 发 生 以 下 情况 中 的 一 种 。 
4 如 果 应 用 处 于 后 台 ， 则 通知 回调 方法 会 被 调用 。 
€ 如 果 应 用 处 于 停止 状态 ， 则 application:didFinishLaunchingWithOptions: 方法 的 
launchOptions 参数 中 的 通知 对 象 (NSDictionary) 是 可 用 的 。 
你 可 能 已 经 注意 到 ， 生 命 周 期 并 不 明确 ， 它 高 度 依赖 于 应 用 的 状态 ， 这 就 使 得 代码 中 的 多 
个 地 方 都 需要 处 理 程序 块 。 
用 于 处 理 通知 的 典型 方式 与 例 5-6 中 所 示 代 码 类 似 。 
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例 5-6 处 理 通知 


-(void)application:(UIApplication *)application 
didReceiveRemoteNotification:(NSDictionary *)userInfo { @ 


// 查 看 下 一 个 方法 


- (void)application:(UIApplication *)application 
didReceiveRemoteNotification:(NSDictionary *)userInfo 
fetchCompletionHandler:(void (^)(UIBackgroundFetchResult))completionHandler (€) 
// 处 理 远 程 通知 一 应 用 正在 运行 @ 
if(application.applicationState == UIApplicationStateInactive) { @ 

// 用 户 点 击 通知 中 心 的 通知 或 报警 弹 窗 
[self processRemoteNotification:userInfo]; 
} else if(application.applicationState == UlApplicationStateBackground) { 
// 应 用 在 后 台 , 不 存在 用 户 交 互 一 一 只 是 获取 数据 
} else { 
// 应 用 已 经 处 于 激活 状态 一 显示 应 用 内 的 更 新 
























































} 
} 


-(void)application:(UIApplication *)application 
didFinishLaunchingWithOptions:(NSDictionary *)launchOptions ( @ 
id notification - [launchOptions 
objectForKey:UIApplicationLaunchOptionsRemoteNotificationKey]; 


if(notification != nil) { © 
NSDictionary *userInfo = (NSDictionary *)notification; @ 
[self processRemoteNotification:userInfo]; 


} 


-(void)processRemoteNotification:(NSDictionary *)userInfo { @ 


Q iOS 7 回调 。 

@ iOS 8 回调。 如 果实 现 了 回调 ， 它 将 取代 application: didReceiveRemoteNotification 77 
法 。 

Q 此 回调 只 在 应 用 运行 时 才 会 被 调用 ， 否 则 该 回调 不 会 被 触发 。 

O 如 果 应 用 处 于 前 台 ， 最 好 不 要 立即 切换 UI， 首 选 方案 是 在 应 用 中 展示 一 个 恰当 的 提示 
(如 应 用 内 横幅 )。 应 用 处 于 后 台 时 ， 用 户 点 击 通知 后 会 触发 回调 ， 此 时 便 可 随意 切换 
ULT. 

O 所 有 的 “魔术 ”都 藏 在 这 里 。 

Q 检查 UIApplicationLaunchOptionsRemoteNotificationKey 键 是 否 存在 。 

O 如 果 存 在 ， 则 远程 通知 数据 是 有 效 的 。 然 后 处 理 这 些 数据 。 

© 处 理 通 知 的 核心 位 置 。 


以 下 是 管理 生命 周期 时 可 以 遵循 的 一 些 最 佳 实践 ， 有 助 于 应 用 提供 最 佳 的 用 户 体验 。 
。 当 应 用 在 活动 状态 时 收 到 一 条 通知 ,要 么 会 忽略 该 通知 ,要 么 会 显示 恰当 的 提示 。 举例 如 下 。 
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4 对 邮件 应 用 来 说 ， 如 果 收 到 有 关 新 邮件 的 通知 ， 更 新 邮件 文件 夹 的 角 标 即 可 。 如 果 
接收 新 邮件 通知 时 ， 线 程 当前 属于 开启 状态 ， 较 好 的 选择 是 更 新 线程 并 显示 一 个 指 
示 符 表明 新 邮件 可 用 。 同 样 ， 如 果 收 到 的 是 不 同 的 聊天 会 话 新 消息 ， 你 或 许可 以 显示 
一 个 应 用 内 横幅 ， 如 图 5-10 所 示 。 横 幅 位 置 应 保证 不 会 妨碍 用 户 继续 完成 当前 任务 。 

4 对 消息 应 用 来 说 ， 如 有 果 在 当前 开启 的 聊天 会 话 中 收 到 新 信息 通知 ， 直 接 在 内 部 展示 
信息 ， 并 更 新 线程 。( 不 要 对 所 有 信息 都 使 用 推送 通知 ， 因 为 不 能 保证 100% 传送 。) 
































Call Info 





© (9): 9 











图 5-10: 在 不 同 的 聊天 会 话 中 收 到 新 消息 时 ，WhatsApp 显示 应 用 内 横幅 


HE 


应 用 在 非 活 动 状态 时 收 到 通知 ， 一 定 是 因为 用 户 之 前 点 击 了 通知 。 但 此 时 用 户 可 能 
经 在 做 别 的 事情 了 ， 这 种 情况 下 ， 比 较 好 的 办 法 是 推送 一 个 新 页 面 ， 这 个 页 面 上 有 支持 
用 户 直接 返回 之 前 页 面 的 选项 。 用 户 体验 的 准确 定义 取决 于 应 用 和 行为 。 

以 某 一 金融 应 用 为 例 ， 如 果 当 前 屏幕 正在 显示 账户 概览 ， 却 收 到 一 个 有 关 刚 完成 的 交易 
的 通知 ， 此 时 可 以 向 用 户 显示 通知 的 相应 细节 ， 并 提供 “返回 ”导航 让 用 户 直接 回 退 至 
账户 概览 。 

e 当 应 用 在 application:didFinishLaunchingWithOptions: 回调 中 获取 到 通知 对 象 的 详细 
信息 时 ， 直 接 将 通知 相关 的 UI 展示 给 用 户 。 
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需要 注意 的 是 ，application:didReceiveRemoteNotification: 只 有 在 应 用 处 于 前 台 时 才 会 
被 调用 ， 然 而 ， 如 果实 现 了 application:didReceiveRemoteNotification:fetchCompletion 
Handter:， 当 应 用 处 于 后 台 或 还 没有 运行 时 ， 此 回调 也 会 被 触发 。 这 也 就 是 说 ， 此 回调 甚至 
会 启动 应 用 。 这 样 的 通知 被 称 为 静默 推送 通知 。 

同样 需要 引起 重视 的 是 ， 后 一 种 方法 可 能 被 调用 两 次 。 

. 第 一 次 是 收 到 通 知 ， 并 且 payload 字段 包含 一 个 键 为 content- available、 值 为 1 的 键 

值 对 时 。 
。 第 二 次 是 用 户 以 通知 中 心 或 警告 的 方式 和 通知 交互 时 。 


可 以 使 用 应 用 的 状态 来 区 分 两 者 ， 如 例 5-6 所 示 。 


通知 应 该 是 有 意义 的 。payload 字段 应 包含 展示 给 用 户 的 文本 信息 以 及 一 些 数 据 ， 这 些 数 
据 在 回调 中 可 用 来 确定 是 否 需要 触发 后 台 拉 取 。 


5.3.2 ”本 地 通知 
不 同 于 远程 通知 ， 当 应 用 处 于 使 用 状态 时 ， 本 地 通知 不 会 展示 任何 UI, 
因此 会 产生 如 下 问题 


。 如 果 应 用 处 于 使 用 状态 ， 为 什么 你 还 需要 显示 本 地 通知 ? 
。 如 果 应 用 没有 处 于 使 用 状态 ， 它 被 挂 起 了 ， 在 这 种 情况 下 怎样 显示 本 地 通知 ? 


答案 便 是 静默 远程 通知 。 如 果 远 程 通知 payload 字段 的 content-available 属性 值 被 置 为 
1， 这 表明 它 会 告诉 操作 系统 远程 通知 不 应 展示 给 用 户 ， 而 是 必须 直接 传递 给 应 用 。 与 普 
通 的 推送 通知 类 似 ， 如 果 需 要 的 话 ， 这 有 可 能 会 唤醒 应 用 。 


随后 ， 应 用 可 以 处 理 数据 ， 需 要 时 还 会 触发 远程 拉 取 ， 同 时 创建 一 个 本 地 通知 。 这 种 方法 
的 优点 是 ， 当 用 户 与 通知 交互 时 ， uite 已 经 被 下 载 、 处 理 并 以 可 用 的 形式 提供 给 应 
用 。 如 此 一 来 ， 显 示 通 知 细节 的 时 间 会 变 得 较 短 ， 应 用 响应 会 加 速 ， 这 更 能 取悦 用 户 。 

当 应 用 处 于 前 台 ， 或 应 用 处 于 后 台 ， 但 用 户 点 击 了 通知 时 ，application:didReceiveRemote 
ei ei 回调 (如 果 没 有 实现 ， 则 调用 application:didRece 


iveLocalNotification: 回调 ) 会 被 调用 。 但 是 当 用 户 在 通知 中 使 用 自 定义 行为 时 ，applica 
tion:handleActionWithIdentifier:forRemoteNotification:completionHandler: 回调 会 被 调用 。 








































































































同时 使 用 本 地 通知 与 静默 推送 通知 ， 可 以 使 得 应 用 在 下 次 启动 时 响应 和 使 用 
速度 都 更 快 。 

















注 5: Stack Overflow, “didReceiveRemoteNotification:fetchCompletionHandler Not Being Called When App Is in 
Background and Not Connected to Xcode” (http://stackoverflow.com/questions/20741618/didreceiveremoten 
otificationfetchcompletionhandler-not-being-called-when-app-is/20851481420851481). 





应 用 的 生命 周期 | 141 


5.4 后 台 拉 取 


后 台 拉 取 功能 是 在 iOS 7 中 推出 的 ， 该 功能 可 以 较 好 地 从 服务 器 定期 同步 数据 。 要 想 启 用 
后 台 拉 取 功能 ， 需 要 以 下 3 个 基本 步骤 。 


(1) 在 项 目 设置 中 开启 功能 。 

(2) 设 置 刷新 间隔 ， 最 好 在 application:didFinishLaunchingWithOptions 中 完成 。 使 用 
-UIApplication setMinimumBackgroundFetchInterval: 方法 请 求 刷 新 以 指定 的 频率 完成 。 

(3) 实现 应 用 的 application:performFetchWithCompletionHandler: 委托 方法 。 如 果 任 务 没 
有 在 30 秒 内 完成 ， 操 作 系 统 会 调度 执行 频率 较 低 的 方法 去 运行 。 
实践 记录 表明 ， 应 用 一 般 使 用 的 时 间 要 少 得 多 ， 通 常 在 2~4 Pb. SERA NIE Ra Py 
将 30 秒 作为 上 限 。 À 


后 台 拉 取 和 推送 通知 可 以 为 用 户 创造 惊人 的 、 愉 快 的 体验 。 以 下 列举 了 一 些 可 以 创造 深刻 
印象 的 指导 原则 。 


。 使 用 后 台 拉 取 与 服务 器 进行 定期 数据 同步 。 将 它 当 作 你 想 要 执行 的 批量 操作 。 

。 不 要 过 分 依赖 后 台 任 务 执行 的 规律 性 。 操 作 系 统 会 定期 调度 它 ， 但 是 调度 的 时 间 间 隔 却 
是 无 规律 的 。 
白天 ， 这 种 间隔 通常 在 10-20 分 钟 (参见 图 5-11)。 影 响 数 值 变化 的 因素 有 : UIBackg 
roundFetchResultNoData 或 UIBackgroundFetchResultNewData 被 当 作 最 终 响 应 结果 的 频 
率 、 完 成 操作 所 花费 的 平均 时 间 、 网 络 情况 、 预 估 的 可 用 带宽 、CPU 和 可 用 内 存 等 )。 

在 夜晚 ， 间 隔 可 能 会 增加 至 几 个 小 时 。 



















































































15:43:14.388 HPerf Apps[20551:8463048] [W] [performFetchWithCompletionHandler] called 





15:43:14.703 HPerf Apps[20551:8463048] [W] [shouldSaveApplicationState] called 
15:43:14.705 HPerf Apps[20551:8463048] [W] [willEncodeRestorableStateWithCoder] called 
16:03:38.540 HPerf Apps[20551:8463048] [W] [performFetchWithCompletionHandler] called 





16:03:38.769 HPerf Apps[20551:8463048] [W] [shouldSaveApplicationState] called 
16:03:38.771 HPerf Apps[20551:8463048] [W] [willEncodeRestorableStateWithCoder] called 
16:13:14.042 HPerf Apps[20551:8463048] [W] [performFetchWithCompletionHandler] called 






16:13:14.369 HPerf Apps[20551:8463048] [W] [shouldSaveApplicationState] called 
16:13:14.370 HPerf Apps[20551:8463048] [W] [willEncodeRestorableStateWithCoder] called 
16:28:14.453 HPerf Apps[20551:8463048 W performFetchWithCompletionHandler] called 
16:28:14.726 HPerf Apps[20551:8463048] [W] [shouldSaveApplicationState] called 
16:28:14.759 HPerf Apps[20551:8463048] [W] [willEncodeRestorableStateWithCoder] called 















图 5-11: 后 台 拉 取 间 隔 的 追踪 (红色 高 亮 部 分 ) 


。 使 用 推送 通知 唤醒 或 启动 应 用 。 
。 使 用 payload 字段 中 的 content-available = 1， 这 样 通知 处 理 方法 还 可 以 从 服务 器 同步 
数据 。 
尽量 只 同步 与 通知 项 相关 的 项 目 ， 而 不 是 同步 所 有 的 东西 。 














注 6: iOS Developer Library, ^UIApplicationDelegate" (http://apple.co/1eMy Y YO). 
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因为 后 台 拉 取 使 应 用 脱离 了 挂 起 状态 ， 所 以 任何 挂 起 的 队列 都 可 能 恢复 。 如 
果 应 用 的 其 他 层 没有 意识 到 任务 队列 可 能 很 快 又 被 暂停 ， 这 也 许 会 导致 灾难 
VERS AAT 

使 用 NSNotifications 通知 应 用 的 不 同 组 件 终止 任何 正在 进行 的 操作 ， 因 为 应 
用 已 经 被 唤醒 来 执行 后 台 拉 取 操作 ， 并 且 不 和 久 以 后 将 再 次 暂停 (30 秒 为 上 限 )。 
使 用 backgroundSessionConfigurationWithIdentifier 配置 NSURLSessionConfiguration 
对 象 ， 从 而 生成 NSURLSession， 这 种 做 法 有 额外 的 好 处 。 它 可 以 使 用 操作 系统 级 
的 守护 进程 来 管理 那些 需要 在 进程 外 长 时 间 运 行 的 任务 。 当 应 用 关闭 或 月 涡 
以 后 ， 后 台 网 络 会 话 还 可 以 继续 。 

对 非 网 络 的 操作 而 言 ， 你 需要 自行 实现 类 似 的 系统 。 

































































智能 静默 通知 
我 曾经 参与 开发 的 一 款 应 用 有 很 强 的 安全 性 要 求 。 具 体 来 说 ， 登 录 成 功 后 生成 的 访问 
令 牌 有 24 小 时 的 有 效 期 。 在 24 小 时 之 内 ， 应 用 可 以 自动 登录 ， 无 需 用 户 重 新 输入 和 赁 
据 。 但 在 24 小 时 后 ， 会 话 将 过 期 ， 需 要 重新 创建 才 行 。 终 庸 置疑 ， 这 必然 不 能 带 来 
很 好 的 用 户 体验 。 
为 了 解决 这 个 问题 ， 我 们 使 用 后 台 拉 取 来 刷新 会 话 。 
Rmn, AWAM, 添加 了 这 个 功能 以 后 ， 应 用 崩溃 得 更 加 频繁 了 。 调 查 显 示 ， 挂 起 的 
操作 队列 恢复 时 并 不 知道 应 用 很 快 会 被 再 次 挂 起 。 其 中 有 一 个 操作 是 与 服务 器 同步 数 
据 ， 但 因为 被 挂 起 了 ， 所 以 同步 无 法 完成 ， 坷 怪 的 事情 发 生 了 : 由 于 没有 活动 ， 连 接 
超时 了 。 如 此 下 去 ， 服 务 器 很 少 能 接收 到 完整 的 数据 ; 就 算 服务 器 有 数据 回应 ， 客 户 
端 也 很 少 能 处 理 完 整 的 数据 ， 这 就 使 得 应 用 处 于 不 一 致 的 状态 。 
我 们 必须 修复 应 用 的 同步 层 ， 确 保 它 可 以 在 触发 同步 操作 之 前 得 到 应 用 的 状态 。 
再 来 讨论 静默 通知 。 服 务 器 在 晚上 会 发 送 静默 推送 通知 。 这 会 唤醒 应 用 ， 然 后 检查 设 
备 是 否 连接 了 WiFi， 是 否 有 足够 的 电池 (如 果 没 有 ， 是 否 在 充电 过 程 中 ) 。 如 果 上 述 
条 件 都 满足 ， 则 抢先 执行 会 话 刷新 。 
因此 ， 无 论 在 第 二 天 何 时 与 应 用 交互 ， 用 户 都 会 觉得 很 顺畅 。 








5.5 ”小结 

本 章 介 绍 了 应 用 的 生命 周期 ， 讲 述 了 生命 周期 如 何 影响 用 户 对 应 用 的 感知 。 要 想 创建 出 用 
户 喜欢 的 应 用 ， 了 解 这 些 是 至 关 重 要 的 。 

有 上 时， 实际 性 能 不 如 感知 性 能 重要 。 为 了 使 用 户 对 应 用 有 良好 的 感知 ， 聪 明 的 做 法 是 使 用 
静默 通知 和 后 台 拉 取 对 应 用 进行 热 启 动 ， 这 样 可 以 为 下 次 使 用 提前 做 好 准备 。 

至 此 ， 本 章 已 经 介绍 了 各 种 情景 下 的 优化 技术 ， 其 中 包括 首次 启动 、 冷 启动 、 热 启动 以 
及 升级 后 启动 。 相 信 你 应 该 能 够 做 一 些 改动 ， 让 应 用 的 启动 时 间 变 得 更 短 ， 使 用 起 来 更 
加 顺畅 。 
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第 6 章 


用 户 界 面 





解 铃 还 须 系 铃 人 。 
一 一 佚名 


当 与 UI 进行 交互 时 ， 大 部 分 用 户 才 注意 到 性 能 问题 。 如 果 某 个 应 用 在 数据 同步 和 刷新 上 
耗 时 较 长 ， 或 用 户 交 互 不 够 稳定 ， 那 么 应 用 会 被 认为 是 迟钝 的 。 
功 耗 、 网 络 使 用 率 、 本 地 存储 等 因素 对 用 户 来 说 是 不 可 见 的 。 因 此 ， 虽 然 这 些 因 素 是 解决 
性 能 问题 的 要 素 ， 但 UI 却 是 应 用 的 门面 ， 如 果 UI 反 应 迟钝 ， 则 必然 会 直接 影响 用 户 的 
反馈 。 
还 有 一 些 无 法 控制 的 外 部 因素 ， 如 下 。 
。 网 络 
弱 网 环境 会 增加 同步 所 需 的 时 间 。 
e. 硬件 
硬件 越 好 ， 其 提供 的 性 能 越 高 。 与 旧型 号 的 iPhone 相 比 ， 搭 载 新 系统 的 新 iPhone 执行 
速度 更 快 ,应 用 可 以 在 不 同 的 CPU 上 运行 ,这 些 CPU 包括 了 32 位 1.3GHz 到 64 位 1.8GHz 
的 所 有 型 号 ， 同 时 支持 1GB 到 2GB 的 RAM, 
。 存储 
应 用 可 以 在 存储 容量 不 同 的 设备 上 运行 ， 存 储 容量 小 至 16GB ， 大 到 128GB ， 它 们 限制 
了 应 用 在 本 地 离线 缓存 数据 的 规模 。 


应 用 还 可 以 根据 所 处 的 运行 环境 作出 不 同 的 决策 ， 保 证 用 户 交互 的 流畅 性 。 


本 章 主要 讨论 如 何 最 小 化 更 新 UI 所 需 的 时 间 。 阅 读 完 本 章 后 ， 你 应 该 可 以 找到 对 应 的 方 
法 ， 让 自己 的 应 用 以 60 帧 每 秒 的 帧 率 (fps) 和 运行。 这 意味 着 应 用 会 有 16.666 毫秒 来 完成 
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向 下 一 帧 过 渡 的 全 部 操作 。 如 果 执 行 一 条 指令 需要 1x 10” 秒 ， 应 用 在 上 述 时 间 段 大 约 可 
以 执行 1000 万 条 指令 。 换 一 个 角度 来 看 ， 如 果 调 用 一 个 简单 的 、 没 有 任何 操作 的 方法 需 
要 大 约 30 纳 秒 〈 包 括 设置 栈 帧 、 参 数 入 栈 、 执 行 以 及 最 终 清 理 的 时 间 )， 那 么 执行 50 多 
万 个 方法 也 是 绰绰有余 的 。 所 以 在 这 个 时 间 段 能 够 执行 很 多 方法 。 





帧 率 
人 的 眼睛 、 大 脑 界面 以 及 视觉 系统 每 秒 可 以 处 理 10~12 个 独立 的 图 像 ， 并 且 可 以 单独 
感知 它们 。 人 类 视觉 感知 的 闪 值 取决 于 被 测 物 。 
当 看 向 较为 明亮 的 显示 器 时 ， 如 果 一 处 黑暗 持 续 了 16 毫秒 或 更 长 时 间 ， 人 们 就 会 开始 
关注 它 。 
根据 设备 的 性 能 ， 有 一 些 方法 可 以 优化 帧 率 。 例 如 ， 如 果 要 在 一 个 RAM 较 小 的 设备 
上 运行 应 用 ， 那 就 在 内 存 中 载 入 较 少 的 数据 。 另 一 个 例子 是 ， 在 较 慢 的 CPU 上 运行 
时 ， 有 尽量 减少 动画 的 使 用 。 











本 章 将 关注 以 下 部 分 : 

。 视图 控制 器 及 其 生命 周期 
。 视图 演 染 
。 自 定义 视图 
。 布局 

。 应 用 扩展 (widgets) 

。 动画 

。 交互 式 通 知 

我 们 将 深入 研究 这 些 组 成 部 分 ， 探 索 可 以 优化 执行 的 方式 ， 并 学 习 一 些 技巧 ， 以 便 用 户 能 
够 获得 较 好 的 感知 体验 。 


> ria 

6.1 视图 控制 器 

视图 控制 器 就 像 胶水 一样 连接 着 数据 服务 和 视图 。 数 据 服务 不 仅 对 内 存 数据 提供 支持 ， 还 
可 以 从 服务 器 或 者 本 地 数据 库 请 求 或 推送 更 新 。 


视图 控制 器 的 生命 周期 依赖 于 它 的 view 属性 ， 该 属性 决定 了 视图 被 创建 、 展 示 、 删 除 和 销 
毁 的 时 刻 。 图 6-1 展示 了 视图 控制 占 的 生命 周期 。 
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即将 展示 的 视图 控制 器 


视图 控制 器 
已 经 被 创建 ? 


ia 将 要 展示 的 视图 
不 
initWithCoder: Sob 


viewDidload: ^^ F----- 




























An 






eee 


视图 已 经 
被 创建 ? 







viewWillAppear: 
viewDidAppear: 


didReceiveMemoryWarning: }¢- - - - - - - - - - 
didReceiveMemoryWarning: 


didReceiveMemoryWarning: 















didReceiveMemoryWarning: 
didReceiveMemoryWarning: 


& 6-1: 视图 控制 器 的 生命 周期 

















在 应 用 开发 的 最 初 阶段 ， 视 图 控制 器 都 较为 精简 ， 状 态 较 好 。 随 着 时 间 的 推移 ， 这 些 视图 
控制 器 慢 慢 变 成 了 所 有 业务 逻辑 的 垃圾 场 ， 代 码 量 也 增长 至 几 千 行 。 虽 然 逻辑 的 “总 量 ” 
是 不 可 避免 的 ， 但 将 代码 重 构成 短小 、 可 复 用 的 方法 是 很 好 的 主意 。 这 样 不 仅 能 解除 耦 
合 ， 还 可 以 发 现 无 用 的 、 重 复 的 代码 。 


下 面 列举 了 创建 视图 控制 器 时 需要 遵循 的 一 些 较 为 基本 的 最 佳 实践 。 


。 保持 视图 控制 器 轻 量 。 在 MVC 结构 的 应 用 中 ， 控 制 器 只 是 纽带 ， 而 不 是 存放 所 有 业务 
逻辑 的 地 方 。 它 甚至 不 属于 模型 。 业 务 逻 辑 应 该 属于 服务 层 或 业务 逻辑 组 件 。 将 它 放 在 
那里 。 
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通过 被 称 为 行为 委托 或 服务 提供 者 的 对 象 ， 视 图 控制 器 应 该 将 服务 组 件 绑 定 至 视图 ， 这 
些 行为 委托 或 服务 提供 者 的 对 象 最 好 能 被 注入 控制 器 (参见 2.13 节 )。 


不 要 在 视图 控制 嚣 中 编写 动画 逻辑 。 动 画 可 以 在 独立 的 动画 类 中 实现 ， 该 类 接受 视图 作 
为 参数 传 入 ， 这 些 视图 就 是 用 来 运行 动画 的 视图 。 然 后 ， 视 图 控制 器 会 将 动画 添加 至 视 
图 或 转 场 效果 上 。 

有 特殊 用 途 的 视图 可 以 拥有 自己 的 动画 。 例 如 ， 一 个 自 定 义 的 微调 控制 器 会 有 它 自己 的 
动画 。 


使 用 数据 源 和 委托 协议 ， 将 代码 按照 数据 检索 、 数 据 更 新 和 其 他 的 业务 逻辑 进行 分 离 。 
视图 控制 器 只 能 用 来 选择 正确 的 视图 ， 并 将 它们 连接 到 供应 源 。 

视图 控制 器 响应 来 自视 图 的 事件 ， 如 按钮 点 击 事件 或 列表 单元 格 的 选择 事件 ， 然 后 将 它 
们 连接 至 数据 接收 器 

视图 控制 器 响应 来 自 HIERO RAI UI 相关 事件 ， 如 方向 变化 或 低 内 存 警 告 。 这 可 能 会 角 
发 视图 的 重新 布局 。 
不 要 编写 自 定义 的 init 代码 。 为 什么 呢 ? 因为 如 果 视 图 控制 器 被 重新 切换 至 XIB 或 故 
Ax. Jp init 方法 永远 都 不 会 被 调用 。 

不 要 在 视图 控制 器 中 使 用 代码 手工 布局 UI， 也 不 要 在 视图 控制 器 中 实现 全 部 的 UI, TH 
图 创建 和 视图 布局 逻辑 等 操作 。 使 用 nibs 或 者 故事 板 。 
手工 布局 代码 不 会 持续 很 入， 因为 应 用 在 不 断 增长 ， 并 且 设 计 也 在 改变 。 在 重新 设计 方 
面 ， 使 用 Interface Builder 比 根据 像素 坐标 来 手动 编写 代码 更 快 。 


此 外 ， 应 用 可 能 会 在 不 同 大 小 和 形状 的 设备 上 运行 。 要 想 适 应 所 有 的 形状 ， 处 理 方向 变 
化 时 的 旋转 操作 ， 以 及 与 每 两 三 年 就 会 变化 的 设计 范例 保持 一 致 ， 通 过 扩展 自 定义 代码 
来 实现 是 比较 难 做 到 的 。 


同样 ， 如 果 某 一 设计 被 分 在 独立 的 nibs 和 故事 板 ， 你 就 可 以 比较 灵活 地 运行 A/B 测试 ， 
因为 在 不 同 的 约束 之 间 很 容易 选择 最 终 需 要 的 。 


比较 好 的 方式 是 ， 创 建 一 个 实现 了 公共 设置 的 基 类 视图 控制 器 ， 甚 他 视图 控制 器 从 这 里 
这 种 技术 并 非 一 直 可 用 ， 因 为 有 时 可 能 需要 在 应 用 的 不 同 部 分 继承 不 同 的 视图 控制 右 。 
例如 ， 在 联系 人 列表 中 应 该 使 用 UITableViewController， 在 用 户 配 置 文件 中 应 该 选择 
UIViewController, 


但 是 ， a eae UIWebView 中 展示 内 容 ， 那 基 类 视图 控制 器 会 是 一 个 不 

错 的 选择 。 如 果 需 要 显示 含有 隐私 策略 的 URL 或 条 款 和 条 件 页 面 ， 那 么 你 是 没 必 要 继 

但 是 ， eo 户 分 享 的 图 片 或 者 视频 (在 信息 应 用 中 )， 你 可 以 创建 子 
， 在 子 类 中 实现 自 定义 浏览 器 或 对 重 写 的 东西 进行 控制 。 


在 各 视图 控制 器 之 间 ， 使 用 category 创建 可 复 用 的 代码 。 如 果 父 视图 控制 器 不 能 满足 
使 用 (例如 ， 在 应 用 中 需要 不 同 种 类 的 视图 控制 器 )， 那 就 创建 category, JHE category 
中 加 上 自 定义 的 方法 或 属性 。 

如 此 一 来 ， 你 就 不 会 被 限制 只 能 使 用 预定 义 的 基 类 ， 同 时 还 能 得 到 复 用 带 来 的 好 处 。 
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现在 我 们 已 经 了 解 了 编写 视图 控制 器 时 的 一 些 最 佳 实践 。 接 下 来 ， 就 UIViewController 生 
命 周期 的 方法 而 言 ， 我 们 将 探讨 哪些 事情 是 该 做 的 ， 哪 些 事情 是 不 该 做 的 。 


6.1.1. 视图 加 载 
视图 初始 化 时 会 涉及 两 个 方法 
不 知 你 是 否 还 记得 ， 当 添加 一 个 新 的 视图 控制 器 时 ， 通 过 Xcode 生成 的 模板 代码 只 
viewDidLoad 方法 。 当 视图 控制 器 的 view 被 请 求 时 ，loadView 方法 会 被 调用 ,但 因为 它 还 
未 被 创建 ， 所 以 会 是 nil, 

视图 会 通过 以 下 三 种 方式 加 载 : 

。 JA nibs 


。 使 用 故事 板 (使 用 UIStoryboardSegue) 

。 使 用 自 定义 代码 创建 UI 

MR ES loadView 方法 创建 了 自 定 义 UI， 你 需要 牢记 以 下 几 点 。 

。 将 view 属性 设置 到 视图 层级 的 根 上 。 

。 确保 视图 正 被 其 他 的 视图 控制 器 所 共享 。 

。 不 要 调用 [super loadView], 

使 用 此 方法 更 改 视图 状态 ， 尤 其 当 视 图 是 从 mnib 文件 加 载 而 来 时 。 我 们 主要 讨论 
viewDidLoad, 
在 视图 层次 结构 准备 就 绪 之 后 ， 视 图 呈现 给 用 户 之 前 ，viewDidLoad 会 被 调用 一 次 。 可 以 
在 该 方法 中 做 一 些 一 次 性 的 初始 化 操作 。 


需要 在 ViewDidLoad 方法 中 做 的 公共 任务 包含 以 下 几 项 。 


。 配置 数据 源 来 填充 数据 。 
。 为 视图 绑 定 数据 。 
这 是 一 个 值得 商 权 的 任务 。 根 据 不 同 的 使 用 情况 ， 你 可 以 一 次 绑 定 数据 ， 并 提供 一 个 刷 
新 按钮 ， 也 可 以 每 次 调用 viewillAppear 时 绑 定 。 使 用 后 者 的 好 处 是 ，UI 总 是 展示 最 
新 的 数据 ， 缺 点 是 ， 如 果 数 据 更 新 不 频繁 (例如 ， 在 新 闻 应 用 中 ) ， 用 户 每 次 看 到 的 都 
是 不 必要 的 刷新 (例如 ， 当 UITableview 重新 绑 定 时 )。 


。 绑 定 视图 的 事件 处 理 程序 、 数 据 源 的 委托 和 其 他 回调 。 
。 注册 数据 的 观察 者 。 
依据 数据 绑 定 到 视图 时 的 不 同位 置 ， 数 据 的 观察 者 可 能 也 会 发 生变 化 。 


。 从 通知 中 心 对 通知 进行 监控 。 
。 初始 化 动画 。 
在 执行 过 程 中 ， 应 该 尽量 缩短 在 viewDidLoad 方法 上 人 花费 的 时 间 。 具 体 来 讲 ， 将 要 被 泻 染 
的 数据 应 该 是 已 经 可 用 的 ， 或 是 在 其 他 线程 进行 加 载 的 。 在 viewDidLoad 的 完成 中 发 生 的 
任何 延迟 ， 都 将 导致 与 视图 控制 器 相关 的 UI 展示 发 生 延 迟 。 用 户 会 卡 在 应 用 启动 或 前 一 
个 视图 控制 器 中 。 

















loadView 和 viewDidLoad, 
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6.1.2 视图 层级 

展示 出 来 的 UI 是 由 内 套 在 树 形 结构 中 的 各 层次 视图 组 成 的 ， 它 们 的 位 置 受 自动 布局 或 其 
他 编排 方式 的 约束 。 视 图 结构 和 谊 染 包括 以 下 步骤 。 

(1) 构造 子 视 图 。 
(2) 计算 并 提供 约束 。 

(3) 为 子 视 图 递归 地 执行 步骤 1 和 步骤 2. 

(4) 递归 泻 染 。 

由 于 视图 层次 变 得 复杂 ， 因 此 需要 更 长 的 时 间 来 构建 和 这 染 视图 。 考 虑 一 个 简单 的 平面 层 
次 结构 : 

e UILabel 

。 自 定义 视图 
e UIImageView 
。 UILabel 


在 一 个 安装 了 iOS 8.1 的 iPhone 6 上 ， 从 视图 控制 器 加 载 (initwithCoder:) 到 演 染 
(viewWillAppear:) 之 前 的 平均 耗 时 约 为 15 毫秒。 而 且 ， 这 还 没有 考虑 从 磁盘 (故事 板 / 
nib 文件 ) 一 次 性 加 载 布局 的 时 间 。viewDidAppear: 方法 会 因为 过 渡 动 画 的 原因 在 约 300 
毫秒 后 被 调用 。 
图 6-2 和 图 6-3 分 别 显示 了 视图 层级 和 记录 日 志 的 时 间 。 

图 6-2 所 示 的 简单 UT 花费 了 约 15 毫秒 来 加 载 ， 留 给 其 他 操作 约 1.6 毫秒 ， 这 样 才能 实现 
60 帧 每 秒 的 泻 染 。 由 于 UI 变 得 越 来 越 复 杂 ， 并 需要 处 理 更 多 的 数据 ， 因 此 优化 执行 变 得 
越 来 越 重要 。 

如 果 一 个 简单 的 示例 需要 花费 如 此 长 的 时 间 来 加 载 (不 泻 染 ) UI， 只 留 下 大 约 1 EMRE 
成 所 有 其 他 的 操作 ， 那 么 我 们 可 以 推断 出 ， 为 了 实现 60 帧 每 秒 的 泻 染 ， 所 有 操作 必须 在 
16.66 毫秒 内 完成 。 














































































































帧 率 与 丢 帧 

因为 nib/ 故事 板 加 载 的 时 间 和 构建 视图 的 时 间 并 没有 可 以 优化 的 空间 ， 所 以 你 需要 回 
头 看 看 绘图 板 ， 找 出 让 用 户 感 到 不 满 的 因素 。 这 是 因为 应 用 不 能 以 60 帧 每 秒 的 速率 运 
行 还 是 因为 应 用 有 一 些 跳 变 ? 

如 果 应 用 每 秒 丢 1 帧 ， 并 以 SO 帧 每 秒 的 速率 运行 ， 那 么 还 是 顺畅 的 。 但 如 果 前 5 秒 都 
是 以 60 帧 每 秒 的 速率 运行 ， 而 在 第 6 秒 丢 掉 了 5 帧 ， 那 用 户 肯 定 会 注意 到 此 处 发 生 了 
跳 变 。 

虽然 已 经 过 去 几 年 了 ， 且 Andrew Munn 的 文献 Follow up to “Android Graphics True 


Facts," 或 The Reason Android is Laggy (https://plus.google.com/u/0/+AndrewMunn/posts/ 
VDKV9XaJRGS) 大 都 是 关于 Android 的 ， 但 现在 还 是 值得 在 睡 前 阅读 一 下 。 
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€ Chapters View Controller Refresh 


Introduce a delay to simulate long 
processing during view load. 


It's in Settings -> VC Load Delay. 


Custom Label 
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Performance 
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6-2: 视图 层次 





-14 11:28:08.570 HPerf Apps[1788:541924] [I] [VC::initWithCoder]: 0.000410 

-14 11:28:08.571 HPerf Apps[1788:541924] [V] prepareForSegue[HPCVC] destination: ident-segue ch07 al vc cls-HPChap 
-14 11:28:08.572 HPerf Apps[1788:541924] [I] :1setMessage] 

-14 11:28:08.586 HPerf Apps[1788:541924] [I] didAddSubview] time-8750ns for UILabel 

-14 11:28:08.588 HPerf Apps[1788:541924] [I] didAddSubview] time-1000ns for HPCustomLabel 
-14 11:28:08.590 HPerf Apps[1788:541924] [I] didAddSubview] time-750ns for UIImageView 
-14 11:28:08.590 HPerf Apps[1788:541924] [I] didAddSubview] time-583ns for UILabel 

-14 11:28:08.591 HPerf Apps[1788:541924] [I] idAddSubview] time-416ns for _UILayoutGuide 
-14 11:28:08.592 HPerf Apps[1788:541924] [I] didAddSubview] time-791ns for _UILayoutGuide 
-14 11:28:08.594 HPerf Apps[1788:541924] [I] p:viewDidLoad]: 0.024300 

-14 11:28:08.595 HPerf Apps[1788:541924] [I] iewDidLoad]: 0.025468 

-14 11:28:08.595 HPerf Apps[1788:541924] [I] [VC: :viewWillAppear]: 0.025855 

-14 11:28:08.596 HPerf Apps[1788:541924] <HPInst> SCR ViewController 

-14 11:28:08.611 HPerf Apps[1788:541924] [I] [MV::layoutSubviews] time-2420416ns 

-14 11:28:08.615 HPerf Apps[1788:541924] [I] [MV::drawRect] time-1083ns 

-14 11:28:09.124 HPerf Apps[1788:541924] [I] [VC::viewDidAppear]: 0.554471 

-14 11:28:09.126 HPerf Apps[1788:541924] [I] [MV::layoutSubviews] time-74250ns 

















6-3: 记录 时 间 


现在 我 们 的 重点 不 仅仅 是 尽量 在 16 毫秒 内 完成 主线 程 的 任务 执行 ， 同 时 还 要 最 大 限度 地 
减少 丢 帧 数 (更 明确 来 说 是 大 量 丢 帧 )。 
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6.1.3 视图 可 见 性 
视图 控制 器 提供 了 四 个 生命 周期 方法 ， 以 接收 有 关 视 图 可 视 性 的 通知 。 





n 


viewWillAppear: 

当 视 图 层级 已 经 准备 好 ， 且 视图 即将 被 放 入 视图 窗口 时 ， 此 方法 会 被 调用 。 在 即将 展示 
视图 控制 器 或 之 前 入 栈 (modal 或 者 其 他 ) 的 视图 控制 器 弹出 时 ， 这 种 情况 就 会 发 生 。 
在 这 个 时 刻 ， 过 渡 动 画 还 未 开始 ， 视 图 对 终端 用 户 也 是 不 可 见 的 。 不 要 启动 任何 视图 动 
画 ， 因 为 没有 任何 作用 。 

viewDidAppear: 

当 视 图 在 视图 窗口 展示 出 来 ， 且 过 渡 动 画 完成 后 ， 此 方法 会 被 调用 。 

因为 动画 会 耗费 约 300 毫秒 ， 所 以 ， 对 比 viewillappear: 和 viewDidLoad:, viewDidAppear: 
和 viewwillappear: 之 间 的 时 间 差 可 能 会 比较 大 。 

启动 或 恢复 任何 想 要 呈现 给 用 户 的 视图 动画 。 

viewWillDisappear: 

该 方法 表示 视图 将 要 从 屏幕 上 隐藏 起 来 。 这 可 能 是 因为 其 他 视图 控制 器 想 要 接管 屏幕 ， 
或 该 视图 控制 器 将 要 出 栈 。 


你 可 能 会 注意 到 ， 当 此 方法 被 调用 时 ， 没 有 办 法 能 直接 够 判断 这 是 由 当前 视图 控制 器 要 
出 栈 还 是 其 他 视图 控制 器 和 人 栈 导致 的 。 


区 分 的 唯一 方式 是 扫描 当前 视图 控制 器 navigationController 的 viewControllers 属 
性 。 为 此 ， 例 6-1 提供 了 一 个 代码 框架 。 


例 6-1 检测 视图 控制 器 的 入 栈 和 出 栈 
-(void)viewWillDisappear:(BOOL)animated { 
NSInteger index = [self.navigationController.viewControllers 
indexOfObject:self]; 
if(index == NSNotFound) { 
// 即 将 出 栈 ,销毁 
} else ( 


// 只 是 保存 状态 ,暂停 







































































































































































} 


[super viewWillDisappear: animated]; 
} 
在 使 用 segue 和 故事 板 时 ， 展 开 的 segue 是 更 好 的 选择 。 侠 果 ' 的 技术 文档 TN2298 很 好 
地 概述 了 如 何 使 用 展开 的 segue。 
viewDidDisappear: 
当 上 一 个 /下 一 个 视图 控制 器 的 过 渡 动 画 完 成 时 ， 此 方法 会 被 调用 。 正 如 
viewDidAppear:, viewWillDisappear: 事件 也 会 有 约 300 毫秒 的 差 值 。 














ü 





E 1: iOS Developer Library, “Technical Note TN2298: Using Unwind Segues" (http://apple.co/1 HK6LP5), 
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还 记得 图 6-1 中 展示 的 生命 周期 图 吗 ? 再 回 过 头 看， 你 会 发 现 这 些 生命 周期 方法 有 可 能 被 
调用 多 次 ， 这 取决 于 用 户 交 互 。 在 这 些 事件 中 还 有 可 能 出 现 无 限 循 环 的 情况 。 


当 应 用 处 于 后 台 或 前 台 时 ， 与 视图 可 视 性 有 关 的 生命 周期 方法 不 会 被 调用 ， 
而 是 通知 UIAppDelegate。 只 有 应 用 处 于 前 台 时 ， 视 图 控制 器 的 生命 周期 方 
法 才 会 被 调用 ， 且 在 过 渡 期 不 会 被 调用 。 

在 视图 控制 器 中 注册 UIApplicationDidBecomeActiveNotification、UIApp 
licationWillResignActiveNotification 和 UIApplicationWillEnterFore 
groundNotification 通知 来 连接 与 视图 可 见 性 相关 的 回调 。 同 时 要 记得 在 
dealloc 中 注销 。 

















以 下 列举 了 一 些 高 效 使 用 生命 周期 事件 的 最 佳 实践 。 


。 无 需 多 说 ， 不 要 重 写 loadView, 

。 将 viewDidLoad 作为 最 后 的 检查 点 ， 查 看 来 自 数 据 源 的 数据 是 否 可 用 。 如 果 可 用 ， 则 更 
新 UI 元 素 。 

。 如 果 每 次 都 需要 展示 最 新 的 信息 ， 那 么 就 使 用 viewWillAppear: 更 新 UI 元 素 。 
例如 ， 在 某 一 个 消息 应 用 中 ， 如 果 在 聊天 会 话 中 观看 完 共享 视频 后 返回 信息 列表 ， 那 么 
用 户 必然 希望 看 到 最 新 的 信息 。 
但 在 新 闻 应 用 中 ， 你 可 能 不 希望 立即 刷新 列表 ， 以 免 用 户 找 不 到 上 下 文 。 在 后 一 种 情况 
下 ， 列 表 视 图 控制 器 一 般 会 监听 来 自 数据 源 的 事件 ， 同 时 ， 应 尽量 精准 、 尽 量 少 地 更 新 
文章 列表 。 

e 在 viewDidAppear: 中 开始 动画 。 如 果 有 视频 等 流 式 内 容 ， 那 么 就 可 以 开始 播放 了 。 订 
阅 应 用 事件 来 检测 动画 /视频 或 其 他 持续 更 新 视频 的 处 理 是 应 该 继续 还 是 停止。 
不 推荐 在 该 方法 中 用 最 新 的 数据 更 新 UI。 如 果 你 这 样 做 了 ， 最 终 的 效果 是 ， 在 过 渡 动 
画 完成 之 后 ， 用 户 会 过 渡 至 旧 的 UI， 然 后 产生 更 新 。 这 个 体验 不 是 很 友好 。 
话 虽 如 此 ， 但 在 一 些 使 用 案例 中 ， 你 不 得 不 在 viewDidAppear: 中 执行 UI 更 新 。 如 果 用 
户 体验 尚 可 接受 ， 那 就 勉强 这 样 吧 。 

e 使 用 viewillDisappear: 来 暂停 或 停止 动画 。 同 样 ， 不 要 做 其 他 多 余 的 操作 。 

e 使 用 viewDidDisappear: 销毁 内 存 中 的 复杂 数据 结构 。 
也 可 以 在 这 里 注销 与 视图 控制 器 绑 定 的 数据 源 通 知 ， 以 及 与 动画 、 数 据 源 、UI 更 新 有 
关 的 应 用 事件 通知 中 心 。 


如 果 其 他 措施 都 不 能 优化 载 和 时间， 那么 你 可 以 在 应 用 中 增加 一 个 精巧 的 动 
画 。 如 此 一 来 ， 你 将 会 有 数 十 秒 的 额外 时 间 来 完成 任务 ， 让 用 户 在 使 用 应 用 
时 不 会 有 明显 的 延迟 。 需 要 注意 的 是 ， 长 时 间 的 动画 会 着 恼 用 户 ， 可 能 导致 
用 户 流失 。 这 应 该 作为 最 后 的 方案 ， 需 要 慎 用 。 




















































































































6.2 视图 


优化 视图 方面 最 具 挑 战 性 的 部 分 是 ， 很 少 有 普 适 于 所 有 视图 的 技术 。 每 个 视图 都 有 其 独特 
的 用 途 ， 且 大 部 分 的 优化 技术 都 与 特定 的 视图 和 暴露 出 的 API 有 关 。 


在 逐一 讨论 这 些 问题 之 前 ， 我 们 先 回顾 以 下 的 基本 规则 。 

。 尽量 减少 在 主线 程 中 所 做 的 工作 。 任 何 额外 代码 的 执行 都 意味 着 更 高 的 丢 帧 概率 。 过 多 
的 丢 帧 会 导致 不 流畅 。 

。 避免 较 大 的 nibs 或 故事 板 。 故 事 板 很 强大 , 但 整个 XML 在 真正 使 用 之 前 必须 被 加 载 (VO) 

和 解析 (XML 处 理 )。 应 该 最 小 化 故事 板 中 的 单元 数目 。 

如 果 需 要 的 话 ， 创 建 多 个 故事 板 或 nib 文件 。 这 可 以 确保 所 有 的 屏幕 并 非 都 在 应 用 启动 

时 一 次 性 加 载 ， 而 是 根据 需要 进行 加 载 。 这 不 仅 有 助 于 减少 应 用 的 启动 时 间 ， 还 能 降低 

整体 内 存 的 消耗 值 。 

当 nib RRA AA, nib 加 载 代码 需要 执行 几 个 步骤 ， 以 确保 nib 文件 的 对 
象 被 创建 并 正确 地 进行 初始 化 。 


当 加 载 的 nib 文件 中 包含 了 对 图 片 或 声音 资源 的 引用 时 ， nib 加载 代码 会 读 取 实 
际 的 图 像 或 声音 文件 ， 将 其 放 入 内 存 并 缓存 。…… 在 iOS 中 ， 只 有 图 像 资源 存储 
在 被 命名 的 缓存 中 。? 
。 避免 在 视图 层次 结构 中 多 层 嵌 套 。 尽 量 保持 扁平 化 。 和 内 套 是 必要 的 “罪恶 ”， 但 仍然 是 
“罪恶 ” o 
在 层次 结构 的 任何 位 置 添加 视图 时 ， 它 的 祖先 树 节点 会 执行 值 为 YES 的 setNeedsLayout: 
方法 ， 当 事件 队列 正在 执行 时 ， 该 设置 会 触发 LayoutSubviews:。 这 个 调用 代价 较 大 ， 
因为 视图 必须 根据 约束 重新 计算 子 视 图 的 位 置 ， 而 且 在 祖先 树 的 每 一 层 都 会 发 生 这 种 
情况 。 


Twitter 团队 贴 出 了 一 篇 文章 ， 他 们 放弃 了 包含 UIImageView 和 一 些 UILabel 的 混合 视 

图 ， 转 而 创建 了 一 个 自 定义 视图 ， 并 对 drawRect: 方法 进行 了 简单 的 优化 。” 

。 尺 可 能 延迟 加 载 视图 并 进行 重用 。 更 多 的 视图 不 仅 会 导致 加 载 时 间 变 长 ， 还 会 使 泻 染 时 
间 变 长 ， 这 些 会 影响 内 存 和 CPU 的 使 用 。 
如 果 需 要 ， 你 可 以 创建 自己 的 视图 缓存 。 这 可 能 会 高 出 UITableView 和 UICollectionView 
已 经 提供 的 对 单元 格 的 复 用 支持 。 当 视图 不 在 视图 窗口 时 ， 这 些 容器 会 释放 视图 。 如 果 
视图 结构 很 复杂 ， 并 且 比 较 耗 费时 间 ， 那 么 实现 自 定义 的 视图 缓存 是 个 明智 之 举 。 


































































































































































































iE 2: iOS Developer Library, "Resource Programming Guide: Nib Files" (https://developer.apple.com/library/ios/docum- 
entation/Cocoa/Conceptual/LoadingResources/CocoaNibs/CocoaNibs.html#//apple_ref/doc/uid/1000005 1i-CH4-SW9). 
1E 3: Twitter Blog, “Simple Strategies for Smooth Animation on the iPhone" (https://blog.twitter.com/2012/simple- 





strategies-for-smooth-animation-on-the-iphone). 
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如 果 使 用 UIScrollview We? 绝对 是 延迟 加 载 。 当 滚动 到 位 置 6 时 才 载 和 视图， 然后 通 
过 建立 自己 的 视图 缓存 ， 模 仿 UITableView 的 行为 。 将 委托 的 scrollViewDidScroll: Fy 
法 与 contentoffset (avira) 属性 结合 起 来 使 用 ， 以 确定 哪些 视图 需要 泻 染 。 
按照 一 般 惯 例 ， 泻 染 的 元 素 会 超过 视图 窗口 的 屏幕 高 度 ， 以 避免 滚动 过 程 中 出 现 抖动 ， 
因为 滚动 开始 时 ， 这 些 元 素 需 要 迅速 地 被 泻 染 出 来 。 
需要 牢记 的 是 ，UITabtLeview 继承 自 UIScroLLView， 这 意味 着 ， 如 果 UITableView 可 以 
做 智能 视图 缓存 ， 那 自 定 义 代 码 也 可 以 实现 。 
。 对 于 复杂 的 UI 而 言 ， 最 好 使 用 自 定 义 绘图 。 这 样 只 会 触发 一 个 视图 进行 绘制 ， 而 不 是 
多 个 子 视图 ， 同 时 也 避免 了 调用 代价 较 高 的 layoutSubviews 和 drawRect: 方法 。 
此 外 ， 要 避免 使 用 具有 通用 目的 及 功能 丰富 的 组 件 而 带 来 的 消耗 ， 你 可 以 使 用 那些 直接 
实现 了 绘制 方法 的 视图 来 代替 。 
例如 ， 如 果 想 要 显示 纯 文本 ， 你 不 需要 使 用 繁重 的 UILabel ( 见 图 6-4). 
介绍 完 这 些 基 本 规则 后 ， 接 下 来 我 们 将 探讨 一 些 更 常见 的 视图 ， 并 深入 了 解 与 每 种 视图 相 
关 的 性 能 技巧 。 

























































































6.2.1 UILabel 


这 可 能 是 iOS 上 最 常用 的 视图 了 7。 它 虽 然 看 起 来 简单 ， 但 是 演 染 代价 却 不 容 小 般 。 下 列 是 

涉及 的 一 些 复杂 步骤 。 

(1) 使 用 字体 、 字 体 类 型 以 及 要 被 泻 染 的 文本 时 ， 计 算 需 要 的 像素 数目 。 这 是 一 个 消耗 较 大 
的 过 程 ， 应 尽 可 能 少 地 去 做 。 * 

(2) 检查 要 被 泻 染 的 宽度 。 

(3) 检查 numberofLines， 计 算 将 要 展示 的 行 数 。 

(4) sizeToFit 是 否 被 调用 ? 如 果 是 ， 则 计算 高 度 。 

(5) 如 果 sizeToFit 没有 被 调用 ， 检 查 当 前 的 内 容 能 否 在 给 定 的 高 度 下 展示 出 来 。 

(6) 如 果 frame 不 够 ， 使 用 LineBreakMode 确定 隐藏 或 截断 的 位 置 。 

(7) 注 意 其 他 的 配置 选项 ， 如 图 6-4 所 示 〈 例 如 ， 是 纯 文 本 还 是 属性 文本 、 阴 影 、 对 齐 、 自 
动 收缩 等 )。 

(8) 最 后 ， 使 用 字体 、 类 型 及 颜色 来 泻 染 最 终 展 示 的 文本 。 


具体 说 明 每 个 UILabel 是 一 件 工作 量 很 大 的 事情 。 使 用 较 少 的 标签 ， 更 容易 管理 效果 ， 使 
用 较 多 的 标签 ， 你 就 需要 多 留意 这 些 标签 的 创建 、 配 置 和 重用 。 
































注 4: 所 需 的 尺寸 计算 必须 在 主线 程 中 完成 。 
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6-4; UILabel 选项 





如 有 果 将 动态 计算 出 的 标签 宽度 作为 容器 宽度 的 一 小 部 分 ， 那 么 你 要 确保 宽度 可 以 由 某 一 百 
分 比 均匀 分 配 。 例 如 ， 两 个 标签 各 占 容 器 宽度 的 50% ， 那 么 容器 宽度 必须 是 偶数 。 避 免 
width/2 这 样 的 调用 。 如 果 宽 度 很 小 ， 其 他 的 都 不 会 受 影 响 ， 除 了 泻 染 ， 因 为 浑 染 需要 反 
句 齿 ， 这 是 一 个 代价 很 大 的 操作 。 




















6.2.2 UIButton 


按钮 几乎 无 处 不 在 ， 如 导航 控制 器 中 的 导航 按钮 、 消 息 应 用 中 的 “发 送 ”按钮 、 自 定义 表 
单 中 的 “发 送 ”按钮 ， 等 等 。 所 以 ， 除 非 应 用 只 有 动画 和 自 定义 泻 当 ， 人 否则 应 用 中 总 会 有 
一 个 按钮 。 

演 染 按钮 的 方式 有 以 下 四 种 : 

。 使 用 自 定义 文本 的 默认 演 染 

。 全 尺寸 资源 的 按钮 

。 可 变 大 小 的 资源 

。 使 用 CALayer 和 贝 塞 尔 路 径 自 定 义 绘 制 

我 们 不 会 探究 每 一 个 细节 ， 主 要 关注 每 一 个 选项 的 优点 和 缺点 。 第 一 个 选项 是 非常 简单 的 ， 
其 余 的 都 在 Designing for iOS: Taming UIButton (https://robots.thoughtbot.com/designing-for- 
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ios-taming-uibutton) 中 进行 讨论 。 
Be 6-1 列 出 了 使 用 每 个 选项 泻 染 按钮 的 利 与 整 。 
表 6-1: 按钮 泻 染 选项 

































































选项 优点 缺点 
自 定义 文本 最 简单 的 方式 ， 可 直接 使 用 通常 是 比较 呆板 、 训 无 装饰 的 按钮 
全 尺寸 资源 可 自 定 义 的 背景 图 片 打 包 在 应 用 中 ， 导 致 包 变 大 
无 需 代 码 即 可 实现 
可 实现 A/B 测试 一 一 图 像 可 在 运行 实验 
时 下 载 
可 变 大 小 资源 可 自 定 义 的 背景 资源 的 任何 更 改 可 能 都 需要 重新 计算 / 
无 需 代 码 即 可 实现 重 置 UIEdgeInsets ffi 
可 实现 A/B 测试 一 一 图 像 可 在 运行 实验 
时 下 载 
包 大 小 的 增 量 相对 较 小 
使 用 CALayer 和 贝 塞 尔 完全 是 自 定义 绘图 任何 格式 的 更 改 或 升级 都 可 能 需要 更 
路 径 自 定义 绘制 新 应 用 


























你 需要 权衡 这 些 选 项 的 利 浆 ， 并 选择 适合 自己 需要 的 那个 。 按 钮 是 一 个 可 以 被 泻 染 的 简单 
组 件 ， 几 乎 不 需要 其 他 性 能 辅助 。 但 如 果 想 要 使 其 更 加 美观 、 靓 丽 和 精致 ， 你 还 需要 探寻 
更 多 其 他 的 可 用 选项 。 























6.2.3 UllmageView 
每 个 应 用 都 含有 图 像 ， 因 为 图 像 让 应 用 看 起 来 更 加 漂亮 。 


但 在 泻 染 代价 较 大 的 各 种 UI 元 素 中 ， 图 像 首届 一 指 。 它 们 大 多 都 是 固定 的 ， 一 旦 创建 就 
` 会 改变 。 要 想 显 示 图 像 的 变化 ， 就 得 加 载 男 外 一 个 图 像 。iOS 仍旧 不 支持 GIF 动画 。 你 
只 能 创建 animationImages 的 一 个 数组 来 存放 可 以 生成 动画 的 图 片 。 一 些 其 他 的 选择 包括 
了 使 用 自 定 义 编码 或 使 用 第 三 方 库 ， 如 ImageMagick (http:/Avww.imagemagick.org/) 或 
AnimatedGIFImageSerialization (https://github.com/mattt/AnimatedGIFImageSerialization ) 。 


在 使 用 Ul Image 和 UlImageView 时 ， 遵 循 以 下 的 最 佳 实践 可 以 提升 性 能 。 


。 对 于 已 知 的 图 像 , 使 用 imageNamed: 方法 加 载 图 像 。 它 可 以 确保 内 容 只 被 加 载 至 内 存 一 次 ， 
还 可 以 确保 在 多 个 UIImage 对 象 间 改 变 用 途 。 

。 在 使 用 imageNamed: 方法 加 载 包 图片 时 ， 使 用 资源 包 。 如 果 应 用 有 一 堆 图 标 ， 且 每 个 图 
标 都 较 小 时 ,这 种 方式 极其 有 用 。 可 以 随意 地 创建 相关 图 像 ( 即 通常 被 一 起 使 用 的 图 片 ) 
的 多 个 目录 。 

当 iOS 从 磁盘 加 载 时 ， 有 一 个 最 佳 的 缓冲 区 大 小 可 以 用 来 在 单独 的 读 操作 中 加 载 多 个 
片 。 此 外 ， 与 开启 一 个 流 并 从 这 个 流 中 读 取 多 张 图 片 相 比 ， 开 启 多 个 IO 流 会 有 一 些 其 
他 支出 。 一 般 情 况 下 ， 读 取 一 个 32KB 的 组 合 文 件 比 读 取 每 个 2KB、 共 16 个 文件 更 快 。 


但 是 ， 如 果 你 想 加 载 一 个 只 使 用 一 次 的 大 图 像 ， 最 好 谨慎 思考 一 下 ， 考 虑 使 用 


imageWithContentsOfFile: (https://developer.apple.com/library/ios/documentation/ 
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UIKit/Reference/UIImage  Class/index.html£//apple, ref/occ/clm/UIImage/ 
imageWithContentsOfFile:) 代替 资源 目录 和 imageNamed: 方法 ， 因 为 资源 目录 缓存 了 这 
些 图 片 (在 这 个 情况 下 并 不 需要 进行 缓存 )。 

在 我 之 前 开发 的 一 款 应 用 中 ， 团 队 成 员 发 现 ， 在 一 个 包 中 的 资源 目录 初始 化 两 屏 所 需 的 
图 片 ， 可 以 将 初始 加 载 时 间 减 少 大 约 300 毫秒 。 


。 对 于 其 他 图 像 ， 使 用 高 性 能 的 图 像 缓 存 库 。AFNetworking 和 SDNebImage 都 是 可 选 的 强 
大 库 。 
当 使 用 内 存 中 的 图 片 时 ， 确 保 正 确 配 置 了 内 存 的 使 用 参数 。 不 要 使 用 硬 编码 。 让 它 能 够 
自 适 应 一 一 使 用 合理 的 RAM 百分比 可 以 较 好 地 进行 配置 。 


。 裁 入 的 图 像 与 即将 泻 染 的 UIImageView 大 小 相同 。 如 果 被 解析 的 图 像 尺 寸 与 UIImageView 
相同 ， 那 么 你 会 得 到 极 高 的 性 能 ， 因 为 调整 图 像 大 小 是 一 个 耗费 较 大 的 操作 ， 如 果 该 图 
像 被 包含 在 UIScroLLView 中 ， 则 耗费 会 更 大 。 

如 果 图 像 来 自 网 络 下 载 ， 那 么 尽量 下 载 和 视图 大 小 匹配 的 图 像 。 如 果 行 不 通 ， 适 当地 对 

图 片 进行 预 处 理 ， 调 整 其 大 小 。 

。 如 果 需 要 使 用 一 些 类 似 于 模糊 或 色调 的 效果 ， 那 么 可 以 创建 一 份 图 像 内 容 的 副本 ， 在 副 
本 上 施加 效果 ， 然 后 使 用 最 终 的 位 图 创建 所 需 的 UIImage。 如 此 一 来 ， 这 些 附 加 的 效果 
只 会 被 使 用 一 次 ， 如 果 有 需要 ， 原 始 图 像 还 可 以 用 于 其 他 显示 。 

。 无 论 使 用 何 种 技术 加 载 图 像 ， 在 非 主 线程 中 执行 ， 最 好 在 一 个 专用 的 队列 中 执行 。 
尤其 要 在 非 主 线程 中 解压 JPG/PNG 图 像 。 

。 最 后 同样 重要 的 是 , 确定 是 否 真 的 需要 图 像 。 如 果 要 展示 一 个 评分 栏 (http://stackoverflow. 
com/questions/27600288/rating-bar-like-android-in-codename-one), ， 最 好 使 用 直接 绘制 的 自 定 
义 视图 ， 而 不 是 使 用 多 个 图 像 ， 通 过 调整 透明 或 覆盖 来 实现 。 

6.2.4 UlTableView 

无 论 是 在 新 闻 应 用 、 邮 件 应 用 、 照 片 流 ， 还 是 其 他 的 应 用 中 ，UITableView 都 是 最 常用 于 

显示 数据 的 视图 。UITableView 提供 了 一 个 展示 信息 条 的 极 好 选择 ， 这 些 信息 条 既 可 以 是 

同一 类 别 ， 也 可 以 是 不 同类 别 。 

UITableView 绑 定 了 两 个 协议 。 

e  UlTableViewDataSource 

必须 将 dataSource 属性 设置 到 数据 源 上 。 顾 名 思 义 ， 数 据 源 是 指 将 要 填充 至 列表 单元 

格 中 的 数据 源 。 

UITableViewDelegate 

必须 将 delegate 属性 设置 到 委托 上 ， 当 用 户 与 列表 或 单元 格 交互 时 ， 此 处 的 委托 必须 

能 接收 到 回调 。 


这 些 协议 和 UITableView 之 间 的 部 分 逻辑 关系 如 图 6-5 所 示 。 
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UlTableViewDataSource 


numberOfRowslInSection 


numberOfRowslnSection 


numberOfRowslnSection 


UlTableViewDelegate 











6-5; UITableView, UITableViewDataSource 和 UITableViewDelegate 


下 列 是 使 用 UITableView 时 需要 牢记 的 一 些 最 佳 实践 。 
© 在 数据 源 的 tableView:cellForRowAtIndexPath: 方法 中 ,使 用 tableView:dequeueReusa 
bleCellWithIdentifier: 或 tableView:dequeueReusableCellWithIdentifier:forIndexPa 
th: 进行 单元 格 的 重用 ， 而 不 是 每 次 都 创建 新 的 单元 格 。 
单元 格 的 创建 有 性 能 成 本 。 如 果 几 个 单元 格 必须 在 很 短 的 时 间 间 隔 内 被 他 








这 将 造成 双 倍 打击 。 重 用 和 





用 户 滚动 列表 视图 时 ) ， 那 么 这 个 成 本 会 成 倍增 长 。 同 时 ， 单 元 格 超出 范 
元 格 意 味 着 只 存在 泻 染 单元 格 这 一 种 开销 。 

















IE aan, 











围 后 会 被 释放 ， 











。 尽 可 能 避免 动态 高 度 的 单元 格 。 诚 然 ， 已 经 确定 的 高 度 代 表 着 只 需 很 少 的 计算 量 。 如 有 果 
内 容 是 动态 配置 的 ， 那 么 不 仅 需要 计算 高 度 ， 而 且 每 次 视图 要 被 泻 染 时 ， 


也 需要 刷新 和 重新 布局 。 这 是 一 个 很 大 
图 6-6 展示 了 填充 固定 高 度 和 可 变 高 度 
度 的 单元 格 ， 最 好 选择 较 高 的 单元 格 ， 





少 了 计算 量 。 











的 性 能 损失 。 
的 UITableView 的 例子 。 如 果 你 想 要 使 用 可 变 高 
因为 这 只 需要 为 较 少 的 单元 格 计算 高 度 ， 从 而 减 





单元 格 的 内 容 











三 Inbox 81 


Informal Android Develop. © 1:35 pm 
» Tomorrow's Meetup: A waitlist is ava... 


Meetup tomorrow Informal Androi... Meetup 
Hackers and Founders Yesterday 
» 2 Meetups this Friday 

Meetup Meetup Reminders Hack... Meetup 
Silicon Valley Entrepren. Yesterday 
» 2 Meetups tomorrow 

Meetup Meetup Reminders Silico... Meetup 
Meetup Friday 
» New Meetup Group: South Bay Com... 
Meetup New Meetup Group! Sou... Meetup 
Silicon Valley Entrepren. E Friday 


» Thursday: Join 107 Founders & Drea... 
Meetup Thursday Startup Socials... Meetup 
Swift Language User Grou. Si Friday 
» Thursday: Join 103 Swifters at “Thin... 
Meetup Thursday Think Different ... Meetup 


Hackers and Founders E Friday 





SNL Revived Celebrity Jeopardy 
for Its 40th Anniversary. And It 
Was Perfect. 


Read more 


Celebrities Vox.com 








SNL's 40th Anniversary Special: 
15 Best and Worst Moments 


Celebrities TVLine.com Read more 
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图 6-6: UlTableView ( 左 侧 是 固定 高 度 的 单元 格 ， 


右 侧 是 可 变 高 度 的 单元 格 ) 
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如 果 你 真 的 需要 动态 高 度 的 单元 格 ， 那 么 定义 一 个 规则 来 标记 单元 格 是 脏 的 。 如 果 某 个 
单元 格 是 脏 的 ， 计 算 它 的 高 度 并 缓存 。 在 委托 的 tableView:heightForRowAtIndexPath: 
回调 中 继续 返回 缓存 的 高 度 ， 直 到 单元 格 不 再 被 标记 为 脏 。 
如 果 要 被 泻 染 的 模型 是 不 可 变 的 ， 一 个 可 用 的 简单 规则 是 ， 检 查 当 前 被 泻 染 的 模型 是 否 
和 相应 的 indexPath 的 值 一 样 。 如 果 一 样 ， 则 使 用 同样 的 值 泻 染 ， 无 需 进 一 步 的 处 理 。 
如 果 不 一 样 ， 则 重新 计算 值 ， 并 将 新 的 对 象 〈 模 型 ) 附加 至 该 单元 格 。 


当 用 自 定 义 视图 重用 单元 格 时 ， 要 避免 通过 调用 LayoutIfNeeded 每 次 都 对 其 进行 布局 。 
即使 一 个 单元 格 的 高 度 是 固定 的 ， 也 有 可 能 出 现 这 样 的 情况 : 在 单元 格 中 的 独立 元 素 可 
能 会 被 设置 成 不 同 的 高 度 ， 例 如 ，UILabet 支持 多 行内 容 ，UIImageView 可 以 装 入 不 同 大 
小 的 图 像 。 

需要 避免 这 种 情况 。 固 定 每 个 元 素 的 尺寸 。 这 可 以 确保 单元 泻 染 所 需 的 时 间 达 到 最 小 值 。 
避免 透明 的 单元 格子 视图 。 创 建 UITableViewCell 时 ， 尽 量 引入 不 透明 元 素 。 半 透明 或 
透明 元 素 (alpha (KF 1.0 的 视图 ) 很 好 看 ， 但 会 有 性 能 损失 。 

出 于 美学 考虑 ， 你 可 能 仍然 希望 将 aLpha 设置 为 小 于 1.0 的 值 。 那 么 你 就 需要 注意 成 本 了 。 
在 快速 滚动 时 考虑 使 用 界面 外 壳 ( 见 图 6-7)。 当 用 户 快 速 滚动 列表 视图 时 ， 虽 然 使 用 了 
所 有 的 优化 ， 但 视图 的 重用 和 泻 染 仍然 需要 超过 16 毫秒 ， 还 有 可 能 出 现 偶 发 的 丢 帧 现 
象 ， 从 而 导致 不 流畅 的 体验 。 
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图 6-7: 使 用 有 外 壳 的 界面 
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在 这 些 情况 下 ， 使 用 一 个 界面 外 这 是 一 个 较 好 的 选择 ， 外 壳 可 以 被 预先 定义 ， 它 的 唯一 
目的 就 是 告诉 终端 用 户 这 些 部 分 即将 展示 一 些 数 据 。 当 滚动 速度 降低 ， 并 低 于 冰 值 时 ， 
刷新 最 终 的 视图 并 填充 数据 。 


你 可 以 使 用 与 列表 视图 相关 联 的 panGestureRecognizer 属性 获取 速度 值 ( 见 例 6-2). 
例 6-2 列表 视图 的 速度 


-(void)scrollViewDidScroll:(UIScrollView *)scrollView { 
CGPoint velocity = [tableView.panGestureRecognizer 
velocityInView:self.view]; 
self.velocity - velocity; 


























} 


-(UITableViewCell *)tableView: (UITableView *)tableView 
cellForRowAtIndexPath:(NSIndexPath *)indexPath { 
if(fabs(self.velocity.y) > 2000) { 

[ENR A 
} else { 
// 返 回 真正 的 单元 格 
} 














} 


。 避免 渐变 .图像 缩放 以 及 任何 屏幕 外 的 绘制 。 这 些 效果 对 CPU 以 及 图 形 处 理 单元 (GPU ) 
来 说 都 是 消耗 。 


6.2.5 UIWebView 

UIWebView 是 用 于 演 染 未 知 或 动态 内 容 的 最 常见 视 
— LEA ERY HTML 或 web URL, 

虽然 有 些 应 用 可 能 全 部 都 是 原生 的 ， 但 还 是 有 需要 使 用 UIWebView 的 场景 ， 以 下 是 一 些 常 
见 场景 。 


。 任何 应 用 中 的 用 户 登 录 。Spotify、Mint 和 LinkedIn 这 样 的 应 用 使 用 原生 UI 演 染 登录 表 
单 。 但 这 有 一 定 的 限制 。 
例如 ， 如 果 想 使 用 CAPTCHA 筛选 刷 屏 的 机 器 人 ， 你 就 需要 为 所 有 的 格式 (http://www. 
theverge.com/2014/12/3/7325925/google-is-killing-captcha-as-we-know-it) 提供 支持 ， 并 将 
其 打包 至 应 用 中 ， 或 将 用 户 指向 一 个 网 页 登陆 URL， 让 服务 器 生成 任何 需要 的 复杂 UL 


。 在 任何 应 用 中 显示 隐私 政策 或 使 用 条 款 。 因 为 这 些 会 随 着 时 间 变 化 ， 并 且 需 要 大 量 的 格 
式 化 (文本 样式 、 编 号 列表 、 其 他 内 容 的 交叉 引用 )， 使 用 原生 视图 不 是 较 好 的 选择 。 
。 新 闻 或 文章 阅读 器 ， 因 为 大 部 分 的 文章 都 是 为 Web 创建 的 ， 几 乎 都 是 HTML. 
。 邮件 应 用 。 例 如 ， 初 始 邮件 是 HTML 形式 ， 当 呈现 消息 或 跟 帖 ， 以 及 撰写 回复 时 。 
如 果 你 需要 展示 较 小 的 富 文 本 ,使 用 UILabel 的 NSAttributedString, 


无 需 CSS xX JavaScript 的 支持 。 它 是 一 个 有 关联 属性 (如 字体 和 字 间 距 ) 集 
合 的 字符 串 ， 这 些 属性 可 以 用 于 字符 串 中 单独 的 字符 或 某 一 范围 内 的 字符 。 























WRI 








。 通 常情 况 下 ， 你 会 将 web 视图 指向 




























































































使 用 UlWebView 时 ， 请 将 以 下 几 个 最 佳 实践 牢记 在 心 。( 需 要 注意 的 是 ， 关 于 UlWebView 能 做 的 





事情 非常 少 ， 并 间 都 是 关注 性 能 的 ， 相反 ， 此 处 的 重点 是 以 最 恰当 的 方式 展示 HTML 内 容 。) 























UIWebView 可 能 比较 笨重 且 迟 钝 ， 所 以 尽 可 能 复 用 web view。 同 时 ，UIWebView 也 因 内 
存 谍 漏 而 知名 。 因 此 ， 每 个 应 用 的 实例 都 应 该 足够 好 。 

无 论 何 时 想 向 用 户 展示 新 的 URL， 先 将 内 容重 置 为 空 的 HTML。 这 样 就 能 确保 web 
view 不 会 将 之 前 的 内 容 展 示 给 用 户 。 要 想 实现 这 一 功能 ， 在 loadRequest: 方法 后 调用 
loadHTMLString:baseURL: 即 可 。 























附加 一 个 自 定 义 的 UIWebViewDelegate。 实 现 webView:shouldStartLoad WithRequest: 
navigationType: 方法 。 要 留意 URL scheme, 如 果 是 http 或 https 以 外 的 东西 ,需要 注意 : 
应 用 应 该 知道 如 何 处 理 这 种 情况 ， 或 警告 用 户 该 网 站 正 试 图 脱离 应 用 。 

这 是 一 个 较 好 的 做 法 ， 不 仅 能 保证 用 户 不 会 突然 出 现在 另 一 个 应 用 当中 ， 同 时 也 对 恶意 
内 容 进 行 了 防护 ， 尤 其 是 恰巧 要 展示 一 个 未 知 URL 的 内 容 时 一 一 例如 ， 在 邮件 或 消息 
应 用 中 。 


你 可 以 通过 stringByEvaluatingJavaScriptFromString: 方法 创建 一 个 桥 来 连接 应 用 和 
JavaScript， 从 而 在 当前 已 经 加 载 的 web 页 面 执行 JavaScript。 如 果 想 要 调用 原生 应 用 的 
方法 ， 你 可 以 参考 之 前 的 处 理 方法 ， 使 用 自 定义 的 URL scheme, 

实现 委托 的 webView:didFailLoadwithError: 方法 ， 以 保持 对 所 有 可 能 出 现 的 错误 的 紧 
实现 webView:didFailLoadWithError: 方法 来 处 理 特定 的 错误 ， 如 例 6-3 所 示 。 如 果 域 名 
与 NSURLErrorDomain 相等 ， 那 么 NSError 对 象 是 有 不 同意 义 的 。 


























例 6-3 用 UIWebView 处 理 错误 


-(void)webView:(UIWebView *)webView 
didFailLoadWithError:(NSError *)error { 
if([NSURLErrorDomain isEqualToString:error.domain]) { 

switch(error.code) ( 
case NSURLErrorBadURL : 
// 处 理 错误 的 URL 
break; 
case NSURLErrorTimedOut: 
// 处 理 超 时 


break ; 





UIWebView 不 会 通知 任何 的 HTTP 协议 错误 ， 例 如 响应 是 404 或 500 的 错误 。 如 例 
6-4 所 示 ， 你 需要 触发 两 次 调用 ， 第 一 次 使 用 自 定义 的 NSURLConnection 调用 ， 然 
后 是 通过 web view 的 调用 。 你 可 以 提供 一 个 NSURLConnection 的 委托 ， 然 后 实现 
connection:didReceiveResponse: 方法 ， 以 便 获取 响应 的 相关 信息 。 








例 6-4 UIWebView 和 HTTP 错误 
@interface HPWebViewController() «UIWebViewDelegate, 
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NSURLConnectionDataDelegate> 
@property (nonatomic, assign) BOOL shouldValidate; 
Qend 
(implementation HPWebViewController 


-(BOOL)webView:(UIWebView *)webView 
shouldStartLoadWithRequest: (NSURLRequest *)request 
navigationType: (UIWebViewNavigationType) navigationType { 


if(self.shouldValidate) { 
[NSURLConnection connectionWithRequest:request delegate:self]; 
return NO; 


} 


return YES; 


} 


- (void)connection: (NSURLConnection *)connection 
didReceiveResponse: (NSURLResponse *)response { 


NSInteger status = [(NSHTTPURLResponse *)response statusCode]; 
if(status >= 400) { 

// 哇 ! 一 个 错误 

// 展 示警 报 或 隐藏 web view 
) else { 

self.shouldValidate - NO; 

[self.webView loadRequest:connection.originalRequest]; 








不 要 展示 错误 的 网 页 


} 


[connection cancel]; 


} 
@end 


因为 这 种 技术 需要 加 载 网 页 两 次 ， 所 以 是 不 受 推荐 的 。 当 加 载 页 面 时 ， 网 页 视图 是 可 以 
展示 错误 的 。 也 许 就 是 点 击 了 接收 的 某 个 销 息 中 的 一 个 连接 ， 用 户 才 发 起 了 请 求 。 


。 GAT UlWebView 的 容器 应 该 提供 以 下 元 素 。 

导航 按钮 (后 退 和 前 进 )。 

取消 按钮 ， 用 于 取消 当前 正在 加 载 的 页 面 。 

用 于 展示 页 面 标题 的 UILabel。 

用 于 退出 web view 的 关闭 按钮 。 如 果 应 用 (如 混合 应 用 ) 只 有 这 一 个 唯一 的 界面 ， 
则 不 需要 关闭 按钮 。 





+ 9 9 9 9 











混合 应 用 是 通过 UIKit HAA HTML 应 用 ， 特 别 是 UlWebView 或 新 的 WKWebView, 
本 书 不 讨论 混合 应 用 的 性 能 (这 个 话题 本 身 就 可 以 写 一 本 书 )。 























iOS 8 的 新 特性 : Webkit 


iOS 8 的 WebKit 特 性 (https://developer.apple.com/library/ios/documentation/Cocoa/Reference/ 
WebKit/ObjC_classic/index.html#//apple_ref/doc/uid/TP30000745) t UIWebView 的 性 能 
更 好 。 如 果 你 正在 写 一 个 新 的 应 用 ， 最 好 使 用 WKWebView。 但 请 记 住 ， 如 果 你 选择 使 用 
WKWebView, 4 iOS 7 设备 上 需要 回 退 到 UIWebView, 


使 用 WKWebView 的 基本 规则 和 之 前 讨论 的 UIWebView 的 规则 是 相同 的 。 作 为 附注 ， 你 可 
以 下 载 一 个 应 用 (https://itunes.apple.com/app/id928647773?mt=8) 来 测试 这 两 者 之 间 的 
KA, ° 











6.2.6 和 上 自 定义 视图 
在 非 游戏 应 用 中 或 不 以 动画 为 核心 的 应 用 中 ， 从 头 开 始 写 自 定义 视 医 
的 方法 是 ， 使 用 Interface Builder 和 自 定 义 nib 文件 创建 复合 视图 。 


虽然 这 是 一 个 极 好 的 初级 技巧 ， 但 是 ， 一旦 创建 了 更 复杂 的 UI， 或 在 列表 视图 中 使 用 了 复 
合 视 图 ， 那 性 能 下 降 的 问题 就 会 暴露 出 来 。 


Twitter 团队 在 其 应 用 开发 的 早期 遇 到 了 这 个 问题 ， 他 们 气 弃 了 复合 视图 的 使 用 ， 改 为 直接 
绘制 视图 ， 从 而 最 大 程度 地 减少 了 泻 染 视图 所 需 的 消耗 。" 


图 6-8 展示 了 一 条 推 文 的 UI。 

















F 不 常见 。 比 较 常用 


WR] 
$- 




































































Ben Sandofsky @sandofsky 
Ship it! 











6-8: 泻 染 的 推 文 











UI 的 基本 实现 可 能 包含 以 下 内 容 ( 见 图 6-9)。 

(1) UIImageView 作为 头像 图 片 。 

(2) FA UILabel 的 NSAttributedText 展示 用 户 名 称 。 

(3) 推 文 的 内 容 使 用 detectorType = UIDataDetectorTypeLink 的 UITextView， 因 为 可 能 包含 链接 。 
(4) 数据 使 用 UILabel。 








1E5: 免责 声明 : 我 并 不 认可 该 应 用 。 
注 6: Twitter Blog, "Simple Strategies for Smooth Animation on the iPhone" (https://blog.twitter.com/2012/simple- 





strategies-for-smooth-animation-on-the-iphone). 
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图 6-9: 原生 推 文 的 视图 


每 个 视图 使 用 额外 的 合成 负责 核心 动画 。 此 外 ， 每 当 子 视图 改变 时 ， 创 建 具 套 的 层次 会 导 


致 VayoutSubviews 被 调用 多 次 。 





为 了 优化 性 能 ， 通 过 在 一 个 drawRect: 中 绘制 全 部 的 元 素 ， 该 团队 创建 了 一 个 自 定义 视图 ， 


如 图 6-10 所 示 。 








Ben Sandofsky 
| Ship it! 





@sandofsky 








图 6-10: 自 定 义 推 文 视图 一 一 直接 绘制 


假设 有 一 个 邮件 应 用 ， 我 们 需要 在 收 件 箱 中 显示 邮件 的 概览 ， 概 览 应 包含 以 下 细节 : 





。 发 送 者 的 姓名 /邮件 ID 

。 发 送 的 日 期 或 时 间 

。 主题 

。 内 容 的 一 些 片 段 (少数 主要 内 容 ) 


。 提示 邮件 是 新 的 、 已 经 被 阅读 的 ， 还 是 已 经 回复 了 的 标记 


。 选择 多 封 邮件 的 选择 器 〈 也 可 能 只 是 一 个 复 选 


E) 





。 邮件 是 否 有 附件 的 标记 
最 终 的 布局 应 该 与 图 6-11 类 似 。 





TE oe | EE 
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4 - Body Snippet 











图 6-11; BRASS 











下 面 将 以 两 种 样式 〈 复 合 视 图 和 自 定义 视图 ) 创建 该 视图 ， 并 测量 这 两 种 视图 的 性 能 
建 一 个 单元 格 的 时 间 和 这 当 所 需 的 时 间 )。 我 们 还 将 分 析 使 用 每 种 样式 的 其 他 优 缺 点 。 
1. 复合 视图 

创建 复合 视图 很 简单 。 你 只 需要 亿 
以 下 是 你 需要 遵循 的 一 些 步骤 。 
(1) 导航 到 File 一 New 一 File。 

(2) 选 择 iOS—Source—Cocoa Custom Touch, 

(3) 类 名 是 HPMaiLCompositeCeLL， 它 是 UITableViewCell 的 子 类 。 
(4) 勾 选 Also create XIB file 的 选项 。 

(5) 点 击 完 成 。 

添加 四 个 UILabeL、 两 个 UIImage 和 一 个 UIButton 元 素 
构 和 图 6-12 中 的 相 匹配 。 


针对 复杂 视图 ， 在 动画 过 程 中 使 用 视图 光栅 化 ， 包 括 但 不 限于 滚动 。 

通常 情况 下 ， 滚 动 期 间 的 视图 布局 是 不 改变 的 ， 在 动画 过 程 中 ， 将 UIView 
的 Layer 的 shouldRasterize 属性 设置 为 YES， 在 动画 完成 之 后 ， 将 其 设置 
2j NO, ' 

















em 


建 一 个 继承 自 UITableViewCell 的 新 的 视图 类 即 可 。 
































( 创 


， 对 它们 进行 排列 ， 使 得 最 终 的 结 





Label Date 


Subject 
4 | Body Snippet 


HHE 


L1 LI 
gara Sender's Name 
QO 


Subject 
Oo Body Snippet 











图 6-12 复合 视图 一 一 布局 (A) 和 独立 (B) 视图 


2. 直接 绘制 





为 了 创建 直接 绘制 的 自 定 义 视图 ， 我 们 再 次 创建 一 个 继承 自 UITableViewCell 的 新 类 ， 














但 


注 7: Stack Overflow, “When Should I Set layer.shouldRasterize to YES?" (http://stackoverow.com/a/19408290). 
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是 不 勾 选 创建 一 个 XIB 文件 的 选项 。 


我 们 将 这 个 类 命名 为 HPMailDirectDrawCell, 


像 之 前 讨论 的 那样 ， 
元 素 。 

















例 6-5 展示 了 泻 染 所 有 元 素 的 代表 性 代码 。 


例 6-5 直接 绘制 的 自 定义 视图 














/ | HPMailDirectDrawCell.h 
typedef NS ENUM(NSInteger, HPMailDirectDrawCellStatus) { 
HPMailDirectDrawCellStatusUnread, 
HPMailDirectDrawCellStatusRead, 
HPMailDirectDrawCellStatusReplied 


a 


@interface HPMailDirectDrawCell : UITableViewCell 


(property (nonatomic, copy) NSString *email; 

(property (nonatomic, copy) NSString *subject; 

(property (nonatomic, copy) NSString *date; 

(property (nonatomic, copy) NSString *snippet; 

(property (nonatomic, assign) HPMailDirectDrawCellStatus mailStatus; 
(property (nonatomic, assign) BOOL hasAttachment; 

(property (nonatomic, assign) BOOL isMailSelected; 


Qend 


(implementation HPMailDirectDrawCell 


// 履 盖 所 有 的 初始 化 器 一 出 于 简洁 省 略 了 


// 履 盖 drawRect 方 法 
-(void)drawRect:(CGRect)rect { 


{ 


UIImage *statusImage = nil; 
switch(self.mailStatus) { 
case HPMailDirectDrawCellStatusRead: 
statusImage = [UIImage imageNamed:@"mail_read"]; 
break; 
case HPMailDirectDrawCellStatusReplied: 
statusImage = [UIImage imageNamed:@"mail_replied"]; 


break; 

case HPMailDirectDrawCellStatusUnread: 

default: 
statusImage = [UIImage imageNamed:@"mail_new"]; 
break; 


} 


CGRect statusRect = CGRectMake(8, 4, 12, 12); 
[statusImage drawInRect:statusRect]; 





UIImage *attachmentImage = nil; 
if(self.hasAttachment) { 
attachmentImage = [UIImage imageNamed:Q"mail attachment"]; 
CGRect attachmentRect - CGRectMake(8, 20, 12, 12); 
[attachmentImage drawInRect:attachmentRect]; 
} 
} 


UIImage *selectedImage = [UIImage imageNamed: 
(self.isMailSelected ? @"mail_selected": @"mail_unselected")]; 
CGRect selectedRect = CGRectMake(8, 36, 12, 12); 
[selectedImage drawInRect:selectedRect]; 


// 或 者 ,能 够 使 用 Core Graphics 绘 制 矢 量 图 像 
} 


CGFloat fontSize = 13; 

CGFloat width = rect.size.width; 

CGFloat remainderWidth = width - 28; 

{ 
CGFloat emailWidth = remainderWidth - 72; 
UIFont *emailFont-[UIFont boldSystemFontOfSize:fontSize]; 
NSDictionary *attrs = @{ NSFontAttributeName: emailFont }; 


[self.email drawInRect:CGRectMake(28, 4, emailWidth, 16) 
withAttributes:attrs]; 


UIFont *stdFont - [UIFont systemFontOfSize:fontSize]; 

NSDictionary *attrs = @{ NSFontAttributeName: stdFont }; 

[self.subject drawInRect:CGRectMake(28, 24, remainderWidth, 16) 
withAttributes:attrs]; 

[self.snippet drawInRect:CGRectMake(28, 44, remainderWidth, 16) 
withAttributes:attrs]; 


UIFont *verdana = [UIFont fontWithName:@"Verdana" size:10]; 

NSDictionary *attrs = @{ NSFontAttributeName: verdana }; 

[self.date drawInRect:CGRectMake(width - 60, 4, 60, 16) 
withAttributes:attrs]; 


} 
@end 


现在 有 两 个 需要 进行 比较 的 方面 : 运行 时 性 能 和 代码 维护 。 


fij 
与 预计 的 一 样 ， 直 接 绘 制 的 自 定 义 视图 中 的 运行 时 性 能 更 好 。 差 异 是 什么 呢 ? 我 们 来 看 看 
图 6-13 中 的 数字 。 
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.177 
.226 
.232 
.237 
.241 
.245 
.249 
.253 
.257 
.785 
.290 
.341 
.352 
.380 


383 
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.398 
.400 
.401 
.401 
.403 
.403 
.405 
.924 
.649 
.680 
.713 
.730 
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[prepareForSegue] i=ch_08_10_cpsv, row=3 
0]: Time=17675125 ns 
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[cell 10]: 
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[prepareForSegue] i-ch 08 10 cpsv, row=4 
: Time-359041 ns 


[cell 0] 


[cell 
[cell 
[cell 
[cell 
[cell 
[cell 
[cell 


<HPInst> SCR 


: Time-177416 ns 
: Time-199041 ns 
: Time-125625 ns 
: Time-110541 ns 
: Time-215166 ns 
: Time-120583 ns 
: Time-222458 ns 
| Views Custom Composite 
[I] [cell 8]: 
[I] [cell 9]: 
[I] [cell 10]: 
[I] [cell 11]: 


Time-1980958 ns 
Time-1913166 ns 
Time-1937208 ns 
Time=1838375 ns 
Time-1976000 ns 
Time-1944208 ns 
Time-1852250 ns 
Views Custom Composite 
Time-6602208 ns 
Time-9596000 ns 
Time-130708 ns 
Time-79916 ns 


Time-293375 ns 
Time-133250 ns 
Time-59208 ns 
Time-28083 ns 








6-13: 复合 视图 对 比 直接 
8 的 代码 ， 表 6-2 总 结 了 复合 视图 和 直接 绘制 的 任务 时 间 。 





根据 例 6-5 Hy 

表 6-2: 数字 比较 

任务 复合 视图 
首次 初始 化 17.6 毫秒 
后 续 初 始 化 1.8-1.9 毫秒 





滚动 后 的 首次 初始 化 6.6 毫秒 
滚动 后 的 第 二 次 初始 化 ”9.6 毫秒 
0.08~0.13 毫秒 


， 性 能 有 2-20 倍 的 惊人 差异 。 使 用 直接 绘制 时 ， 初 始 加 载 速度 会 快 50 倍 。 
而 且 这 些 数 据 还 是 在 直接 绘图 的 代码 没有 优化 的 情况 下 得 出 的 。 





因此 ， 从 性 


量 级 。 


然而 ， 从 维护 的 角度 来 看 ， 代 码 会 难 
确 地 将 复合 UI 换 成 直接 绘图 























o 








直接 绘制 
0.36 毫秒 


0.1~0.2 毫秒 


0.3 毫秒 
0.13 毫秒 
0.03-0.08 毫秒 


能 角度 来 看 ， 在 某 些 时 候 ， 直 接 





绘制 一 一 比较 初始 化 和 重用 

















绘图 提供 的 性 能 比 复合 视图 提供 的 要 好 一 个 数 





以 维护 和 发 展 。 一 旦 应 用 稳定 下 来 ， 你 就 可 以 比较 明 





此 外 ， 如 果 创 建 的 视图 仅 包括 标准 控件 ， 做 A/B 测试 时 ， 比 较 容 易 的 方式 是 根据 不 同 的 设 
这 样 的 话 ， 你 可 以 展示 不 同 的 布局 ， 而 无 


备 发 送 新 的 nib 文件 ， 然 后 从 nib 文件 加 载 UI。 














需 发 行 应 用 的 新 版 本 。 








6.3 自动 布局 


iOS 6 推出 了 自动 布局 ， 减 轻 了 在 复杂 屏幕 中 调整 元 素 的 痛苦 。 通 过 被 称 为 约束 的 东西 ， 自 
动 布局 可 以 描述 视图 相互 之 间 的 位 置 、 容 器 、 大 小 等 。 约 束 可 以 描述 一 个 元 素 距 另 一 元 素 的 
距离 〈 水 平 或 垂直 )、 其 大 小 〈 宽 度 或 高 度 ) ， 或 其 与 另 一 元 素 的 对 齐 方式 EREE). 


这 里 假设 你 已 经 具备 了 自动 布局 的 知识 。 如 果 没 有 ， 你 应 该 回顾 iOS 开发 者 库 的 官方 参考 
(https://developer.apple.com/library/ios/documentation/UserExperience/Conceptual/AutolayoutPG/) 
或 Matthijs Hollemans 的 Beginning Auto Layout Tutorial in iOS 7: Part 1 (http://www. 
raywenderlich.com/50317/beginning-auto-layout-tutorial-in-ios-7-part-1 ) 。 


虽然 自动 布局 是 一 个 不 错 的 方法 ， 它 允许 你 将 元 素 的 位 置 、 大 小 交 给 核心 引擎 ， 而 不 是 在 
代码 中 实现 ， 但 这 却 带 来 了 性 能 开销 。 自 动 布局 的 实现 涉及 了 求解 线性 方程 组 ， 这 些 方 
程 组 就 是 为 了 满足 约束 而 列 出 的 。 它 使 用 Cassowary (http://stacks.11craft.com/cassowary- 
cocoa-autolayout-and-enaml-constraints.html) 约束 求解 工具 包 。 作 为 任何 通用 方程 的 解答 
器 ， 它 的 复杂 度 为 O(N)， 其 中 入 是 约束 的 数目 ， 而 不 是 元 素 的 数量 。 这 意味 着 ， 在 一 般 
情况 下 ， 为 了 确定 视图 中 所 有 元 素 的 位 置 和 大 小 ， 可 能 有 大 概 AN 个 方程 要 求解 ， 而 且 解 
方程 花费 的 时 间 与 元 素 的 个 数 和 包含 的 约束 个 数 是 不 成 比例 的 。 

在 Florian Kugler 的 实验 中 ， “如 果 视 图 的 数量 增加 至 几 百 个 , 自动 布局 需要 几 十 秒 或 更 多 的 
时 间 ， 而 直接 设置 结构 大 小 在 毫秒 级 别 就 可 完成 。 一 般 情 况 下 ， 直 接 设 置 结构 大 小 要 比 使 
用 自动 布局 快 1000 倍 左右 。Kugler 的 测试 结果 如 图 6-14 所 示 。 你 可 以 在 Github 上 找到 用 
来 测试 的 应 用 源 代码 (https://github.com/oriankugler/AutoLayoutProling ) 。 




















































































































水 平 的 视图 层级 
20 
o 手动 设置 由 l 
o 自动 布局 (相对 于 父 视图 的 位 置 ) 
自动 布局 (彼此 的 相对 位 置 ) 

















B 6-14: 自动 布局 性 能 





注 8: Florian Kugler, “Auto Layout Performance on iOS" (http://floriankugler.com/2013/04/22/auto-layout-performance- 





on-ios/). 
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比较 有 趣 的 是 ， 自 动 布 局 使 用 本 地 约束 (例如 ， 元 素 彼此 之 间 的 位 置 关 系 ) 会 比 使 用 全 局 
约束 更 快 ， 其 中 全 局 约束 指 相对 于 父 类 视图 的 位 置 。 

任何 自 定义 代码 将 会 对 视图 有 专门 的 了 解 ， 如 此 一 来 ， 与 使 用 通用 的 方程 解法 来 确定 视图 
的 位 置 和 大 小 相 比 ， 这 种 泻 染 的 速度 更 快 。 

话 虽 如 此 ， 当 Kugler 对 更 多 真实 的 应 用 进行 测量 时 ， 测 量 数字 表明 ， 虽 然 自 动 布 局 较 慢 ， 
但 在 测试 应 用 中 慢 的 程度 没有 超过 10 倍 或 100 倍 。 例 如 ， 在 图 6-15 中 所 展示 的 应 用 中 ， 
自动 布局 花费 了 大 约 180 毫秒 ， 而 自 定义 代码 花费 了 大 约 120 毫秒 。 所 以 ， 虽 然 自动 布局 
需要 50% 额外 的 时 间 ， 但 它 仍 然 不 是 一 个 十 分 糟糕 的 选择 。 
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B 6-15: 测试 自动 布局 的 真实 应 用 


对 自动 布局 的 最 终 裁 决 是 什么 ? 靠 自 己 的 判断 。 衡 量 展示 视图 和 演 染 视图 所 需 的 时 间 。 如 
果 超过 了 国 值 ， 考 虑 使 用 自 定义 代码 。 国 值 根据 具体 的 应 用 而 定 。 请 记 住 ， 使 用 自动 布局 
时 ， 布 局 性 能 和 浑 染 性 能 总 会 有 提高 的 空间 。 

就 自 定义 代码 而 言 ， 每 次 布局 更 新 时 ， 总 有 搬运 和 测试 代码 的 负担 。 这 也 意味 着 ， 你 可 能 
需要 放弃 使 用 自 定义 模板 运行 A/B 测试 的 方案 。 


6.4 ”尺寸 类 别 


直到 iPhone 4S， 应 用 的 开发 都 是 简单 的 ， 因 为 只 需要 为 一 种 大 小 和 分 辨 率 进行 开发 。 
iPhone 5 和 5S 的 垂直 尺寸 都 有 所 增加 。iPhone 6 和 6 Plus 在 水 平和 垂直 方向 都 增加 了 更 
多 的 像素 。 在 iPad 系列 中 ，iPad 4 在 上 一 代 的 基础 上 将 分 辩 率 增加 了 一 倍 。 每 英寸 的 像素 
(pixels per inch, ppi) 增长 了 3 倍 ， 因 此 ， 应 用 的 设计 人 员 和 开发 人 员 需 要 将 两 倍 的 图 像 
进行 打包 (这 使 得 包 的 下 载 更 难 )， 更 别提 直接 绘图 时 的 庞大 像素 数量 了 。 



































6-3; iOS 设 备 一 一 屏幕 分 辩 率 和 密度 

















设备 屏幕 分 辨 率 ”像素 密度 (ppi) 
iPhone 3G 320 x 480 163 
iPhone 4 640 x 960 326 
iPhone 4S 640 x 960 326 
iPhone 5 640x1136 326 
iPhone 5S 640x1136 326 
iPhone 6 750x1334 326 
iPhone 6 Plus 1080 x 1920 401 
iPad 2 1024 x 768 132 
iPad (第 三 代 ) 2048x1536 264 
iPad ( 第 四 代 ) 2048 x 1536 264 
iPad Air 2048 x 1536 264 
iPad Air 2 2048 x 1536 264 
iPad Mini 1024 x 768 163 


iPad Mini (Retina) 2048 x 1536 325 


iOS RRR REAS, ESA TC A ER, TER, ppi 不 是 点 到 像素 的 比例 。 
将 点 当 作 iOS 提供 的 一 个 缩放 因素 ， 那 么 你 就 不 用 担心 缩放 本 身 了 。 因 此 ， 一 个 10pt 视图 
可 能 对 应 iPhone 3G 的 10 像素 、iPhone 4 和 5S 的 20 像素 ， 以 及 iPhone 6 Plus 上 的 30 像 
素 。 定 义 约束 时 是 以 点 为 单位 的 ， 如 图 6-16 所 示 。 字 体 大 小 同样 以 点 为 单位 。 
































Constraints + Font System 17.0 2 
-| | c a 
| Font System - System 
Constant: = 10 "i | - 
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Priority: 1000 
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ai Multiplier: 1 TA 176 
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Edit Done 
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Autoshrink | Fixed Font Size E 











& 6-16; 视图 约束 


当 使 用 直接 绘图 的 方式 创建 自 定义 视图 时 ，drawRect: 方法 给 出 了 大 规模 的 CGRect 尺寸 。 
针对 之 前 创建 的 自 定义 单元 格 ， 你 需要 注意 : 在 iPhone 5S 和 iPhone 6 上 展示 的 大 小 是 
320x64. 


苹果 公司 没有 向 开发 者 暴露 确切 的 像素 尺寸 ， 取 而 代 之 地 做 了 一 件 非常 出 色 的 事情 ， 即 通 
过 所 谓 的 尺寸 类 别提 取出 配置 。 尺 寸 类 别 标识 了 要 展示 的 高 度 和 宽度 的 相对 量 。 























注 9: 硬件 像素 。 软 件 像素 在 461 ppi 上 实际 是 1242 x 2208 (http://bit.ly/curious-case-iphone6)。 
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视图 控制 器 的 可 用 尺寸 类 别 基于 以 下 三 个 因素 

。 屏幕 尺寸 

。 设备 方向 

。 视图 控制 器 可 用 的 部 分 屏幕 (注意 ， 当 使 用 拆 分 的 视图 控制 器 展示 主 从 控制 器 时 ， 没 有 
任何 控制 器 可 以 访问 整个 屏幕 ) 

在 interface builder 中 设计 视图 控制 器 时 ， 底 部 的 布局 工具 条 附近 有 尺寸 类 控制 器 ， 使 用 它 

选择 一 个 类 ， 如 图 6-17 所 示 。 
































L 


Compact Width | Any Height 











F t * 4 
I—À( * 4 
Base Values 
For all compact width layouts 
(e.g. 3.5-inch, 4-inch, and 4.7-inch 





iPhones in portrait or landscape) 


w Compact h Any 


iOS 定义 了 两 种 尺寸 类 : 紧凑 型 和 普通 型 。 当 使 用 有 限 空 间 时 ， 选 择 紧 凑 型 的 尺寸 ;使 用 
广阔 空间 时 ， 选 择 普通 型 。 如 图 6-17 所 示 ， 你 可 以 选择 一 个 水 平 尺 寸 类 和 垂直 尺寸 类 来 配 
置 最 终 的 UI。 

针对 不 同 设备 以 及 不 同 的 方向 组 合 ，iOS 开发 者 库 中 的 文章 iOS Human Interface Guidelines 
(https://developer.apple.com/library/ios/documentation/UserExperience/Conceptual/MobileHIG/ 
LayoutandA ppearance.html#//apple_ref/doc/uid/TP40006556-CH54-SW1) 提供 了 一 些 可 选择 
的 尺寸 类 的 详细 信息 。 图 6-18 将 设备 和 方向 与 尺寸 类 进行 了 映射 。 

对 于 任何 视图 而 言 ， 你 都 能 够 更 改 所 有 可 能 影响 最 终 位 置 和 大 小 的 参数 。 其 中 包括 所 有 视图 
中 的 约束 ， 以 及 所 有 可 用 的 字体 大 小 。 图 6-19 展示 了 如 何 对 某 一 特定 的 尺寸 类 添加 约束 。 

















6-17; 尺寸 类 选择 
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图 6-18: 尺寸 类 与 设备 /方向 的 映射 
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图 6-19. 配置 尺寸 类 一 一 具体 的 参数 
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尺寸 类 提供 了 对 UI 进行 分 类 的 不 同方 法 ， 无 需 再 为 简单 场景 (如 方向 ) 创建 独立 的 布局 。 
尽量 在 各 个 场景 使 用 它们 一 一 建立 基础 设施 的 全 部 原因 就 是 为 了 简化 开发 。 

从 性 能 角度 看 ， 加 载 故事 板 或 XIB 文件 只 有 极 少 的 影响 。 比 较 好 的 部 分 是 ， 这 些 基 于 类 的 
约束 对 最 终 的 XIB 文件 的 贡献 只 是 一 小 部 分 。 此 外 ， 如 果 应 用 支持 多 个 方向 ， 那 XIB x 
件 只 被 加 载 一 次 ， 当 方向 发 生变 化 时 ， 这 为 应 用 提供 了 很 大 的 援助 。 而 且 请 注意 ， 在 应 用 
运行 时 加 载 XIB 文件 是 一 次 性 消耗 。 




















尺寸 类 需要 自动 布局 。 如 果 你 因为 性 能 原因 不 选择 使 用 自动 布局 ， 那 就 不 能 
使 用 尺寸 类 。 











6.5 iOS 8 中 新 的 交互 特性 


iOS 8 推出 了 两 个 神奇 的 功能 ， 可 以 让 用 户 与 应 用 交互 : 
。 交互 式 通知 
。 应 用 扩展 


以 下 两 市 将 探索 这 些 功能 。 我 们 先 查 看 每 个 特性 的 基本 设置 ， 然 后 看 看 如 何 使 用 它们 提供 
最 佳 的 用 户 体验 。 


6.5.1 交互 式 通知 

你 可 以 使 用 交互 式 通 知 ， 从 而 允许 用 户 提 供 一 个 针对 输入 的 快速 响应 。 

截止 至 1OS 7， 用 户 点 击 通知 (或 在 锁 屏 上 请 动 ) 只 会 调 起 应 用 。 后 续 的 操作 只 能 在 应 用 
中 进行 。 在 iOS 8 中 ， 开 发 者 为 用 户 提供 了 一 些 操作 ， 这 些 操 作 是 在 通知 中 预定 义 的 行为 。 
以 用 户 请 动 的 那个 通知 为 基础 ， 确 定 将 要 在 应 用 中 执行 的 行为 。 在 application:didFinish 
LaunchingWithOptions: 方法 被 调用 时 ， 通 知 的 值 可 以 从 Launchoptions 中 获取 到 。 

这 一 种 方式 很 好 ， 但 却 无 法 处 理 以 下 的 两 种 情况 : 其 一 ， 某 个 通知 期 望 用 户 能 够 响应 ， 且 
同时 存在 多 个 可 用 的 响应 选择 ， 其 二 ， 获 取 响 应 的 速度 比 点 击 后 启动 应 用 、 将 用 户 引导 至 
某 一 具体 的 视图 控制 器 (是 一 个 非常 麻烦 且 耗 时 的 过 程 ) 更 快 。 

在 iOS 8 中 ， 你 可 以 为 一 个 通知 添加 类 别 ， 这 样 就 可 以 为 该 类 别 定制 一 个 或 多 个 动作 。 用 
户 可 以 选择 其 中 任何 可 用 的 动作 。 

下 面 的 例子 是 使 用 交互 式 通知 时 的 可 能 动作 。 

。 邮件 : 回复 ， 标 记 为 垃圾 。 

E 提醒 ， 回复 。 

。 在 社交 应 用 中 评论 信息 : 回复 评论 ， 点 先 评 论 。 

。 任务 和 提醒 : 稍 后 ， 标 记 完 成 。 


























































































































图 6-20 显示 了 一 个 示例 ， 在 使 用 交互 式 通知 时 ， 开 发 者 提供 了 用 于 作出 回应 的 快速 操作 界 
面 。 此 处 的 例子 是 一 个 提醒 通知 。 当 向 左 滑动 时 ， 向 用 户 展示 了 两 个 操作 选项 ， 稍 后 和 完 
成 。 如 果 用 户 选 择 稍 后 ， 则 提醒 通知 过 一 段 时 间 后 会 再 次 弹出 。 如 果 用 户 选择 了 完成 ， 则 
任务 被 标记 为 完成 。 
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6-20: 交互 式 通知 一 一 任务 提醒 








这 样 做 的 好 处 是 ， 用 户 不 需要 打开 应 用 就 可 以 采取 进一步 的 行动 。 在 实现 了 UIAppDelegate 
协议 的 类 中 ，application:handleActionWithIdentifier:forLocalNotification: 回调 会 处 
理 通知 。 


在 使 用 下 拉 横 幅 登 录 设备 时 ， 用 户 也 可 以 响应 通知 。UI 有 一 些 细小 的 差别 ， 如 图 6-21 所 示 。 
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6-21; 交互 式 提醒 横幅 


6.5.2 ”应 用 扩展 
苹果 开发 者 网 站 (https://developer.apple.com/app-extensions) 对 应 用 扩展 进行 了 如 下 描述 。 
通过 iOS 8 和 OS X Yosemite， 应 用 扩展 使 用 户 能 够 获取 应 用 的 功能 和 信息 。 例 
如 ， 你 的 应 用 可 以 作为 部 件 在 “今日 ”屏幕 上 展示 ， 在 操作 表 中 可 以 添加 新 的 按 
钮 ,在 iOS 照片 应 用 内 提供 照片 过 滤器 ， 或 显示 一 个 新 的 全 系统 自 定 义 键盘 。 使 
用 扩展 将 应 用 的 强劲 能 力 放 在 用 户 最 需要 的 地 方 。 
iOS 8 引入 了 可 以 添加 至 应 用 的 应 用 扩展 。 图 6-22 显示 了 一 个 Xcode 菜单 ， 在 此 可 以 将 新 
的 扩展 添加 至 主 应 用 中 。 
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g 
5 Choose a template for your new target: 
1 





] ios A 
| Application job = KZ ee 
Framework & Library 
RESHENINENENTSE Action Custom Document Photo Editing 
PP Extension Keyboard Provider Extension 
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OSX 出 17 
Application = 
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Application Extension Extension 





System Plug-in 
Other 
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6-22: 添加 应 用 扩展 的 Xcode 菜单 


iOS 8 中 可 用 的 应 用 扩展 如 下 。 


。 今日 (通常 被 称 为 窗口 小 部 件 ) 
通知 中 心 的 “今日 ”视图 可 以 帮助 用 户 快速 更 新 或 快速 完成 某 一 任务 ( 见 图 6-23)。 











Notifications 


Sunday, 


® Weather! 


YAHOO! 


Afternoon: 














6-23. Yahoo 天 气 应 用 的 窗口 小 部 件 
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自 定义 键盘 
允许 用 户 在 任何 应 用 中 使 用 自己 喜欢 的 键盘 ， 包 括 搜索 ( 见 图 6-24) 。 自 定义 键盘 最 终 
能 够 自由 地 取代 iOS 系统 键盘 ， 并 在 所 有 应 用 中 使 用 ， 这 一 行为 无 疑 是 伟大 的 。 
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6-24. 自 定义 键盘 


A 

允许 用 户 更 加 无 颖 地 跨 应 用 共享 数据 。 深 层 链接 同样 允许 共享 数据 ， 但 会 强迫 用 户 改 为 
目标 应 用 的 上 下 文 。 

行动 

帮助 用 户 查 看 或 改变 在 主 应 用 中 发 起 的 内 容 。 例 如 ， 在 收 到 视频 时 ， 你 可 能 希望 在 喜爱 
的 媒体 播放 器 中 观看 ， 而 不 是 在 实际 收 到 视频 的 电子 邮件 7 消 息 应 用 中 观看 。 

照片 编辑 

允许 用 户 编辑 照片 应 用 中 的 照片 或 视频 。 所 以 ， 你 现在 可 以 下 载 一 个 像 Adobe 
Photoshop Express 的 应 用 来 编辑 所 有 的 照片 ， 而 无 需 在 该 应 用 中 手动 打开 要 被 编辑 的 照 
片 ， 你 现在 可 以 从 照片 中 启动 编辑 器 了 。 

文档 提供 者 

允许 其 他 应 用 访问 自己 应 用 所 管理 的 文件 。 对 某 一 类 特殊 类 型 的 文档 而 言 ， 文 档 提供 者 
扮演 了 本 地 仓库 的 角色 ， 让 用 户 将 所 有 文件 收集 至 一 个 地 方 。 


这 些 扩展 被 当 作 副 包 打包 至 主 应 用 中 ， 由 同一 生命 周期 和 应 用 开发 的 规则 管理 。 
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扩展 特有 的 最 佳 实践 仍 在 不 断 发 展 。 可 能 需要 经 历 两 个 完整 的 应 用 版 本 的 发 行 ， 才 能 将 
模式 和 最 佳 实践 进行 合并 。 目 前 ， 大 部 分 准则 和 各 个 iOS 应 用 的 准则 没什么 差别 。 


6.6 ”小结 


对 视图 控制 器 生命 周期 有 了 深入 的 理解 后 ， 现 在 你 可 以 调整 应 用 的 感知 性 能 了 。 用 户 可 能 
知道 应 用 缓存 数据 是 为 了 获得 更 快 的 加 载 时 间 ， 或 是 对 网 络 消耗 的 重视 ， 但 如 果 最 终 的 用 
户 交互 执行 地 很 差 ， 那 这 一 切 都 变 成 次 要 的 了 。 

在 声明 性 UI 和 程序 性 UI 之 间 做 选择 时 ， 你 应 该 以 性 能 和 可 扩展 性 为 依据 。 本 章 对 常见 视 
图 的 深入 分 析 应 该 使 你 能 明智 地 使 用 它们 ， 而 对 于 应 用 中 的 给 定 场景 ， 自 定义 视图 的 禁 代 
方案 也 为 你 提供 了 相关 的 选择 。 
需要 注意 的 是 ， 当 以 60 帧 每 秒 的 速率 渲染 或 更 新 UI 元素、 展示 动画 时 (An. iius 
他 方式 )， 你 大 约 有 16 毫秒 来 执行 所 有 需要 的 操作 。 这 可 能 包括 网 络 或 磁盘 VO, WEA 
容 的 更 新 、 布 局 和 最 终 的 演 染 。 可 以 按照 下 述 原 则 将 任务 拆 分 为 子 任务 ， 这 个 原则 是 : 在 
一 个 事件 循环 中 ， 保 证 子 任务 在 主线 程 上 累积 消耗 的 时 间 最 短 。 



































在 应 用 中 使 用 网 络 是 必 不 可 少 的 ， 但 减少 网 络 延 迟 的 方法 却 是 有 限 的 〈 例 如 ， 使 用 CDN 
或 边缘 服务 器 、 使 用 Protobuf 或 数据 压缩 这 样 小 型 的 payload 字段 格式 )， 因 此 ， 你 应 该 着 
手 对 网 络 条 件 进 行 最 大 程度 的 优化 ， 并 预先 对 不 同 的 场景 进行 规划 。 


本 章 将 重点 关注 影响 整体 延迟 的 因素 ， 并 讨论 如 何 充分 利用 现 有 的 信息 来 最 大 程度 地 提高 性 能 。 


7.1 指标 和 测量 


在 网 络 中 完成 的 大 多 数 工作 是 你 无 法 控制 的 ， 因 此 确定 衡量 的 标准 非常 重要 。 我 们 将 在 本 
章 中 讨论 一 些 较为 重要 的 性 能 相关 指标 。 请 注意 ， 此 处 并 非 要 列举 出 所 有 的 指标 ， 只 是 挑 
出 在 性 能 优化 相关 的 测量 中 更 为 重要 的 一 些 指标 。 


图 7-1 展示 了 典型 的 网 络 请 求全 景 图 。 









































DNS 查询 
CDN/ 边 缘 服 务 器 解决 方案 
SSL 提 手 
请 求 
响应 
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& 7-1: 网 络 一 一 设备 至 服务 器 
后 续 讨 论 的 大 致 结构 是 : 先进 行 相关 指标 的 说 明 ， 然 后 是 一 个 或 多 个 示例 ， 最 后 是 最 佳 实践 。 
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7.1 
发 起 


用 变 


.1 DNS 查 找 时 间 


连接 的 第 一 步 是 DNS 查找 。 如 果 你 的 应 用 严重 依赖 网 络 操作 ，DNS 的 查找 时 间 会 使 应 





gj 


慢 。 在 一 个 关于 两 个 位 置 的 统计 样本 中 ， 从 加 利 | 





























尼 亚 州 的 森 尼 韦 尔 市 访问 www.google. 








com 时 ，DNS 查找 时 间 是 2846 毫秒 ， 而 从 印度 的 新 德里 访问 时 只 花费 了 34 毫秒 ( 见 图 7-2)。 


查找 
由 成 
使 用 
7-2 H 














时 间 与 主 DNS Jl oS ERER RARR. AAE IR] ERE 





内 容 分 发 网 络 (content delivery network, CDN) 将 延迟 最 小 化 是 一 种 常见 的 做 法 。 在 图 


到 目的 卫 地址 的 路 

















它 被 角 











Fh， 你 应 该 注意 到 www.google.com 在 两 个 地 点 解析 的 他 地 址 是 不 同 的 。 E 森 尼 韦 尔 市 ， 
坚 析 成 了 美国 的 一 台 服 务 器 ， 而 在 新 德里 ， 它 被 解析 成 了 印度 的 服务 器 。 但 

















KA DNS 











会 为 每 一 个 独 有 的 子 域名 进行 查找 ， 所 有 ， 拥 有 多 个 CDN 主机 名 会 导致 应 用 | m 





束 度 变 慢 
XR EAD, 








e A^ — bash — 80x25 
^» dig www.google.com 


; <<>> DiG 9.8.3-P1 <<>> www.google.com 

;; global options: +cmd 

;; Got answer: 

;; -»»HEADER««- opcode: QUERY, status: NOERROR, id: 37677 

;; flags: qr rd ra; QUERY: 1, ANSWER: 6, AUTHORITY: 0, ADDITIONAL: 0 


;; QUESTION SECTION: 
;www.google.com. 


;; ANSWER SECTION: 

www.google.com. 140 
«google.com. 140 
.google.com. 140 
«google.com. 140 
«google.com. 140 
.google.com. 140 


; Query time: 286 msec 

; SERVER: 75. UE J5, 75#53 (75. 75.75.75) 
; WHEN : mEMEEMERS FERE 

; MSG SIZE rcvd: 128 


— bash — 79x23 
^» dig www.google.com 


; <<>> DiG 9.8.3-P1 <<>> www.google.com 

;; global options: +cmd 

77 Got answer: 

j; —>>HEADER<<- opcode: QUERY, status: NOERROR, id: 28929 

;; flags: qr rd ra; QUERY: 1, ANSWER: 1, AUTHORITY: 0, ADDITIONAL: 0 


;; QUESTION SECTION: 
; www.google.com. 


77 ANSWER SECTION: 
www.google.com. 66 216.58.196.68 


;; Query time: 34 msec 

; SERVER: 192.168.1.1453(192.168.1.1) 
; WHEN: y rem -— 

; MSG SIZE rcvd: 48 











7-2: www.google.com 的 DNS 查询 时 间 一 一 加 州 的 森 尼 韦 尔 市 (上 ) 和 印度 的 新 德 


里 UM 
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第 7 章 





为 了 最 大 限度 地 减少 DNS 查询 时 间 所 产生 的 延迟 ， 你 应 该 遵循 以 下 的 最 佳 实践 。 


。 最 小 化 应 用 使 用 的 专 有 域名 的 数量 。 





最 好 是 能 做 到 以 下 几 点 : 


(1) 身份 管理 (登录 、 注 销 、 配 置 文件 





— 


(2) 数据 服务 (API 端点 ) 
(3) CDN (图 片 和 其 他 静态 人 工 产品 ) 


有 可 能 需要 其 他 域名 〈 例 如 ， 用 于 提供 





























按照 路 由 的 一 般 工作 方式 , 多 个 域名 是 不 可 避免 的 。 


视频 、 上 传 检测 数据 、 有 具体 的 子 数据 服务 、 广 告 


投放 ， 甚 至 是 国家 特定 的 全 球 本 地 化 )。 如 果子 域名 数量 上 升 至 两 位 数 ， 那 么 势必 会 引 


发 担忧。 




















。 在 应 用 启动 时 不 需要 连接 所 有 的 域名 ， 可 能 只 需要 身份 管理 和 初始 画面 所 需 的 数据 。 对 
于 后 续 的 子 域名 , 尝试 更 早 地 进行 DNS 解析 , 也 被 称 为 DNS 预先 下 载 。 为 实现 此 操作 ， 


你 可 以 参考 以 下 两 点 。 
如 果子 域名 和 主机 在 控制 范 





























返回 HITP 204 的 状态 码 ， 然 后 提前 对 该 URL 发 起 连接 。 
第 二 个 方法 是 使 用 gethostbyname 执行 一 个 明确 的 DNS 查找 。 然 而 ， 针 对 不 同 的 协议 ， 
主机 可 能 会 解析 至 不 同 的 卫 ， 例 如 ，HTTP 请 求 可 能 会 解析 至 一 个 地 址 ， 而 HTTPS 会 
解析 至 另 一 个 地 址 。 虽 然 不 是 很 常见 ， 但 第 7 层 的 路 由 可 以 根据 实际 的 请 求解 析 卫 地 
像 是 一 个 地 址 ， 视 频 是 另外 一 个 地 址 。 鉴 于 这 些 因素 ， 在 连接 之 前 解析 
DNS 经 常 是 无 用 的 ， 对 主机 进行 伪 连 接 会 更 有 效 。 


7.1.2 SSL 握手 时 间 
为 了 安全 起 见 ， 可 以 假设 应 用 中 所 有 的 连接 均 是 通过 TLS/SSL 的 (使 用 HTTPS)。HTTPS 


址 ， 例 如 ， 








a 











在 连接 开始 时 ， 先 进行 SSL 握手 ，SSL 握手 主要 是 验证 服务 器 证 
随机 密 钥 。 这 一 操作 听 起 来 简单 ， 但 是 却 有 很 多 步 又， 还 会 耗费 较 多 时 间 CL 

















围 内 ， 你 可 以 配置 一 个 预 设 的 URL， 不 返回 任何 数据 ， 只 


图 7-3), 





上 攻 ， 同 时 共享 用 于 通信 的 

















客户 端 问候 消息 

服务 器 问候 消息 
服务 器 证 书 
Jat E 
共享 密 钥 回复 N 
A 


客户 端 结束 


























服务 器 结束 





7-3: SSL 握手 


你 应 该 遵循 以 下 的 最 佳 实践 。 


。 最 大 程度 地 减少 应 用 发 起 的 连接 数 。 因 此 ， 也 需要 减少 应 用 连接 的 独 有 域名 的 数量 。 

















。 请 求 结束 后 不 要 关闭 HTTP/S 连接 。 
为 所 有 的 HTTPS 请 求 添 加 头 Connection: keep-alive。 这 确保 了 同样 的 连接 在 下 一 
请 求 时 可 以 复 用 。 

。 使 用 域 分 片 。 如 此 一 来 ， 虽然 连接 的 是 不 同 的 主机 和 名， 你 也 可 以 使 用 同一 个 socket， 只 
要 它们 解析 为 相同 的 也 ， 可 以 使 用 相同 的 证 书 例如， 在 通配符 域 ) 就 行 了 。 

域 分 片 在 SPDY 及 其 后 续 版 本 一 一 HTTP/2 (https://http2.github.io) 中 是 可 用 的 。 你 
需要 一 个 支持 上 述 任意 一 种 格式 的 网 络 库 。 


iOS 9 对 HTTP/2 有 原生 的 支持 。 
对 于 iOS 8 和 之 前 的 版 本 ， 你 需要 一 个 第 三 方 库 ， 如 Twitter 的 CocoaSPDY 
你 可 以 从 GitHub (https://github.com/twitter/CocoaSPDY) 上 获取 。 通 过 引入 
CocoaPod CocoaSPDY 进行 使 用 。 
































7.1.3 ”网络 类 型 


由 于 用 户 逐 步 委 弃 了 桌面 设备 ， 这 也 就 放弃 了 总 是 处 于 连接 状态 的 高 速 带宽 网 络 ， 转 而 使 
用 有 同等 质量 的 WiFi 网 络 或 使 用 间歇 连接 的 带宽 可 变 的 移动 网 络 。 更 有 挑战 的 场景 是 ， 
用 户 是 移动 的 。 当 设备 在 移动 信号 塔 之 间 切 换 时 ， 网 络 和 质量 也 会 发 生变 化 。 一 个 设备 可 
能 在 任何 时 刻 从 LTE 网 络 切 换 到 GPRS 或 进入 无 信号 区 域 。 对 于 这 种 情况 ， 人 们 往往 束 手 
无 策 。 
































主机 的 可 到 达 性 
你 可 以 使 用 革 果 公司 的 可 到 达 性 库 (https://developer.apple.com/library/ios/samplecode/ 
Reachability/Introduction/Intro.html) 或 使 用 Tony Million 对 该 库 的 简易 替换 (https:// 
github.com/tonymillion/Reachability) ， 主 机 的 可 到 达 性 发 生变 化 时 ， 蔡 换 的 库 支 持 回调 
的 调用 。 











如 果 设 备 闲 置 超过 几 秒 (具体 的 值 是 不 确定 的 ， 这 里 的 儿 秒 也 可 能 变 成 几 分 
钟 )， 网 络 无 线 电 可 能 已 经 关闭 ， 这 将 导致 无 线 资源 控制 器 发 生 额外 的 几 百 
或 儿 千 毫秒 的 延迟 。 

先 确定 主机 的 可 到 达 性 ， 从 而 确保 应 用 具备 处 理 此 种 场景 的 能 











一 般 情 况 下 ，iPhone $H iPad 可 以 使 用 下 列 任何 网 络 连接 到 互联 网 。 

。 WiFi 
如 果 WiFi 网 络 是 私有 网 络 (如 家 庭 或 办 公 室 连接 )， 那 么 你 可 以 期 望 连接 至 互联 网 的 网 
络 是 持续 且 质 量 较 好 的 。 
但 是 ， 处 于 WiFi 网 络 中 并 不 能 保证 一 定 连接 上 了 互联 网 。 例 如 ， 当 设备 连接 到 某 一 公 
共 热 点 〈 例 如， 在 旅馆 或 购物 商场 ) 时 ， 如 果 用 户 没有 成 功 地 提供 适当 的 赁 证， 那么 将 
无 法 访问 互联 网 。 









































即使 设备 已 经 连接 至 互联 网 ， 也 可 能 有 一 些 限 制 ， 比 如 可 连接 的 域名 或 端口 限制 。 举 
个 例子 ，www.google.com 或 www.yahoo.com 域 可 能 是 允许 的 ， 但 mail.google.com 或 
mail.yahoo.com 却 可 能 无 法 使 用 。 


4G: LTE、HSPA+ (高 速 的 数据 网 络 ) 

这 些 是 最 新 一 代 的 数据 网 络 。 在 第 一 个 真正 的 业务 数据 发 送 前 ， 一 般 会 有 100-600 毫秒 
的 延迟 。 这 些 网 络 以 亚 毫 秒 的 间隔 动态 地 创建 无 线 相关 的 资源 ， 并 且 爆 发 性 地 发 送 数据 。 
理论 上 来 说 ， 速 度 会 从 100Mbps 浮动 至 1Gbps。 对 于 高 速率 移动 的 通信 ， 如 在 汽车 或 
火车 上 ， 速 度 可 能 会 在 D00Mbps; 对 于 低速 率 移动 的 通信 ， 如 行人 或 静止 的 用 户 ， 速 度 
可 能 会 达到 1Gbps。 

3G: HSDPA, HSUPA, UMTS, DMA2000 (中 等 速度 的 数据 网 络 ) 

这 些 是 上 一 代 的 数据 网 络 ， 但 使 用 频率 可 能 比 LTE 更 频繁 。 

3G 的 速度 可 能 会 从 200Kbps 变化 至 超过 50Mbps。 双 向 的 传输 速度 可 能 并 不 对 称 。 
HSDPA RARAHI FINE, HSUPA 具有 较 高 的 上 行 速度 。 

2G: EDGE, GPRS (低速 的 数据 网 络 ) 

20 世纪 90 年 代 的 网 络 仍 然 没 有 消亡 。 这 些 都 是 初始 数字 网 络 (第 1 代 网 络 使 用 模拟 信 
号 )， 提供 了 较 低 的 带宽 。EDGE 理论 上 具有 500Kbps 的 极限 速度 ， 而 GPRS 最 高 只 能 
达到 50Kbps。 


























可 到 达 性 库 可 以 给 出 访问 主机 的 详细 网 络 信息 。 使 用 此 信息 来 确定 传输 内 容 的 类 型 〈 例 
如 ， 文 本 、 图 像 或 是 视频 ) ， 以 及 传输 的 各 种 项 目的 大 小 ， 等 等 。 











0.2% 的 数据 传输 ; 46% 的 功 耗 ! 
2011 年 ， 密 欢 根 大 学 和 AT 以 芽 发 布 了 “移动 应 用 性 能 资源 使 用 情况 ”(http://mobility 
first.winlab.rutgers.edu/documents/mobisys11.pdf) ， 这 篇 研究 性 论文 分 析 了 移动 应 用 的 网 
络 使 用 和 功 耗 效率 。 
论文 探讨 了 潘多拉 公司 ， 将 其 在 移动 网 络 的 间 歌 性 网 络 传输 的 低 效 率 问 题 作 为 经 典 案 
例 研 究 。 虽 然 问题 已 经 修复 了 ， 但 案例 分 析 还 是 值得 阅读 的 。 
当 播 放 一 首 歌 曲 时 ， 应 用 会 将 这 首 歌 全 部 下 载 ， 这 是 正确 的 行为 : 下 载 尽 可 能 多 的 数 
据 ， 让 无 线 电 关闭 的 时 间 尽 可 能 
但 是 ， 在 传送 之 后 ， 应 用 会 每 60 秒 定期 地 发 送 检 测 事件 。 这 些 事件 仅 占 传 输 总 字 节 的 
0.2%, [gp T ÈM E 723685 46%, 
事件 的 数据 通常 是 比较 小 的 ， 但 因为 无 线 电 在 较 长 的 时 间 都 会 保持 激活 状态 ， 所 以 它 
将 应 用 的 电量 消耗 增加 了 一 倍 。 
通过 将 这 些 数据 分 发 至 一 些 请 求 之 中 ， 或 在 无 线 电 已 经 处 于 激活 状态 时 再 发 送 数据 ， 
就 可 以 消除 不 必要 的 能 量 拖 尾 ， 实 现 更 高 的 电源 效率 。 








为 确保 你 的 应 用 不 会 成 为 类 似 案 件 研 究 的 一 部 分 ， 在 开发 以 网 络 为 中 心 的 应 用 时 ， 你 可 以 











遵循 以 下 的 最 佳 实践 。 























设计 时 考虑 不 同 的 网 络 可 用 性 。 在 移动 网 络 中 ， 唯 一 不 变 的 是 ， 网 络 可 用 性 是 多 变 的 。 
对 于 流 媒体 ， 最 好 选择 HTTP 实时 流 或 任何 可 用 的 自 适 应 比特 率 流 媒体 技术 ， 这 些 技术 
可 以 在 某 一 时 刻 针对 可 用 带宽 进行 动态 切换 ， 切 换 至 当前 带宽 的 最 佳 流 质量 ， 从 而 提供 
流畅 的 视频 播放 。 

对 于 非 流 媒 体内 容 ， 你 需要 实现 一 些 策略 ， 确 定 在 单 次 拉 取 时 应 该 下 载 多 少数 据 ， 并 且 
数据 量 必须 是 自 适 应 的 。 例 如 ， 你 可 能 不 希望 在 最 新 一 次 更 新 时 ， 一 次 拉 取 所 有 的 200 
封 新 邮件 。 你 可 以 先 下 载 前 50 封 邮 件 ， 再 逐步 下 载 更 多 邮件 。 


同样 ， 在 低速 网 络 时 ， 不 要 打开 视频 自动 播放 功能 ， 这 可 能 会 花费 用 户 很 多 钱 。 


对 于 自 定义 的 非 流 媒体 数据 拉 取 ， 要 保持 对 服务 器 的 关注 。 让 客户 端 发 送 网 络 特征 数 ， 
服务 器 决定 返回 的 记录 条 数 。 这 样 一 来 ， 你 可 以 在 不 发 布 应 用 新 版 本 的 情况 下 进行 适应 
性 改变 。 

出 现 失败 时 ， 在 随机 的 、 以 指数 增长 的 延迟 后 进行 重 试 。 

例如 ， 第 一 次 失败 后 ， 应 用 可 能 会 在 1 秒 后 重 试 。 第 二 次 失败 时 ， 应 用 在 2 秒 后 重 试 ， 
接着 是 4 秒 的 延迟 。 不 要 忘记 对 每 个 会 话 设置 最 多 的 自动 重 试 次 数 。 

设立 强制 刷新 之 间 的 最 短 时 间 。 当 用 户 明确 要 求 刷新 时 ， 不 要 立即 发 出 请 求 。 相 反 ， 检 
查 是 否 已 经 存在 一 个 请 求 ， 或 当前 请 求 与 上 次 请 求 的 时 间 间 隔 是 否 小 于 浆 值 。 如 果 满 足 
上 述 条 件 ， 则 不 要 发 送 此 次 请 求 。 

使 用 可 到 达 性 库 发 现 网 络 状态 的 变化 。 如 图 7-4 所 示 ， 使 用 指示 条 向 用 户 展示 不 可 用 的 
状态 ， 毕 况 设 备 没有 网 络 连接 并 非 你 的 错 。 通 过 让 用 户 了 解 潜在 的 连接 问题 ， 可 以 避免 
你 的 应 用 受到 指责 。 
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Warning 
Seems you are offline. Please check 
your network connection. 
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& 7-4; Spotify, Facebook X0 TOI 针对 离线 网 络 状 态 的 指示 条 











不 要 缓存 网 络 状态 。 不 论 是 通过 触发 请 求 时 的 回调 来 获取 状态 ， 还 是 在 发 送 请 求 之 前 显 


式 地 检查 状态 ， 


要 始终 使 用 网 络 敏感 度 高 的 任务 的 最 新 值 。 























基于 网 络 类 型 下 载 内 容 。 如 果 想 要 展示 一 个 医 





应 该 始终 下 载 和 设备 适 配 的 图 像 
别 很 大 。 








Z 





像 , 不 用 总 是 下 载 原 始 的 、 高 质量 的 图 














fi. 


iPhone 4S 所 需 的 图 像 尺寸 和 第 三 代 iPad 所 需 的 差 


如 果 应 用 有 视频 内 容 ， 最 好 有 一 个 与 之 关联 的 预览 图 像 。 如 有 果 应 用 支持 自动 播放 功能 ， 
































在 非 WiFi 网 络 中 只 展示 预览 图 像 ， 因 为 自动 播放 会 花费 用 户 很 多 钱 。 
此 外 ， 针 对 图 像 、 音 频 和 视频 等 ， 提 供 一 个 关闭 自动 下 载 或 关闭 自动 播放 功能 的 选项 。 
图 7-5 展示 了 WhatsApp 应 用 中 相关 的 设置 。 
eseo: 9 91% NO oeoo 9 91% ao 
< Settings Data Usage € Data Usage Images 
MEDIA AUTO-DOWNLOAD 
Never 
Images Wi-Fi and Cellular 
Wi-Fi 
Audio Wi-Fi 
Wi-Fi and Cellular v 
Videos Wi-Fi 
Documents Wi-Fi 


Reset Auto-Download Settings 


Voice Messages are always automatically downloaded for the 
best communication experience. 


CALL SETTINGS 
Low Data Usage 


Enabling this option further lowers the amount of data used 
during a WhatsApp call. 


Network Usage 


Chats 











7-5; WhatsApp 中 对 图 像 、 音 频 和 视频 内 容 下 载 的 可 选 设置 


乐观 地 预先 下 载 。 在 WiFi 网 络 中 预先 下 载 用 户 在 后 续 时 刻 需要 的 内 容 。 随 后 就 可 以 使 
用 缓存 内 容 了 。 最 好 分 次 下 载 内 容 ， 在 使 用 之 后 关 掉 网 络 连接 ， 这 有 助 于 节省 电量 。 





没有 黄金 法 则 可 循 。 这 在 很 大 程 
数 、 预 期 的 使 用 模式 和 网 络 条 件 。 








预先 下 载 永远 都 是 争论 的 焦点 。 在 下 载 最 少数 据 和 获取 最 近 可 能 需要 使 用 的 
所 有 内 容 之 间 ， 人 们 总 会 存在 激烈 争执 。 





度 上 取决 于 平均 数据 的 大 小 、 完 成 的 下 载 
如 果 网 络 不 断 变化 ， 而 且 你 需要 执行 最 小 


的 数据 传输 ， 看 看 是 否 可 以 分 批 处 理 这 些 请 求 。 








如 有 果 适 用 ， 当 网 络 可 用 时 ， 支 持 同步 的 离线 存储 。 通 常情 况 下 ， 网 络 缓存 就 足够 了 。 但 
如 有 果 需 要 更 多 的 结构 化 数据 ， 使 用 本 地 文件 或 Core Data 会 是 一 个 较 好 的 选择 。 
对 游戏 来 说 ， 缓 存 最 近 一 级 的 详细 信息 。 对 邮件 应 用 来 说 ， 存 储 一 些 带 有 附件 的 最 新 电 
子 邮件 是 一 个 不 错 的 选择 。 
根据 不 同 的 应 用 ， 你 可 能 会 允许 用 户 创建 新 的 离线 内 容 ， 离 线 内 容 会 在 网 络 连接 可 用 时 
和 服务 器 进行 同步 。 例 如 ， 在 邮件 应 用 中 编写 新 邮件 或 回复 某 封 邮件 时 ， 在 社交 应 用 中 
更 新 资料 图 片 时 ， 拍 摄 将 要 上 传 的 照片 或 视频 时 。 

总 是 将 网 络 和 通信 与 UI 解 辜 。 如 果 应 用 可 以 进行 离线 操作 ， 那 么 就 通知 用 户 离线 操作 
是 可 行 的 ， 否 则 就 通知 用 户 不 能 进行 离线 操作 。 不 要 让 用 户 已 经 开始 和 应 用 进行 交互 之 
后 ， 获 取 不 到 返回 值 。 这 会 是 一 个 糟糕 的 用 户 体验 。 












































不 要 为 金融 、 银 行 、 股 票 交 易 或 其 他 需要 和 服务 器 同步 的 应 用 增加 离线 交易 
的 选项 ， 因 为 更 新 的 数据 可 能 在 离线 模式 下 不 可 用 。 














图 7-6 展示 了 离线 模式 下 的 Facebook 和 E*Trade 应 用 。Facebook 应 用 通知 用 户 网 络 不 
可 用 ,但 允许 发 布 评论 或 状态 更 新 。 当 网 络 可 用 时 ， 上 述 内 容 会 在 后 续 进 行 同 步 。 在 
E*Trade 应 用 中 ， 用 户 也 可 以 与 应 用 进行 交互 ， 但 当 查 找 某 一 股票 的 报价 上 时， 应 用 会 进 
入 死胡同 ， 这 会 导致 糟糕 的 用 户 体验 。 
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图 7-6: 离线 模式 下 的 应 用 一 一 Facebook (Zr) 和 E*Trade (4) 
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需要 注意 的 是 ， 网 络 条 件 总 是 会 超出 应 用 的 控制 ， 但 在 这 些 限制 范 上 














内 提供 的 用 户 体验 却 





pun 











是 受 应 用 控制 的 。 要 将 可 用 选项 做 到 最 好 ， 这 些 选项 包括 离线 存储 、 网 络 可 到 达 性 、 网 络 


类 型 、 执 行 


(或 不 执行 ) 网 络 操作 ， 通 知 (或 不 通知 ) 用 户 相关 的 信息 。 


7.1.4 HEIR 


延迟 是 指 从 服务 器 请 求 资源 时 ， 在 网 络 传输 上 花费 的 额外 时 间 。 设 置 用 于 测量 网 络 延迟 的 
系统 是 很 重要 的 。 


网 络 延迟 可 以 通过 使 用 请 求 过程 中 花费 的 总 时 间 减 去 服务 器 上 花费 的 时 间 (计算 和 服务 响 


应 ) 来 测量 : 


m 





Round-Trip Time 
Network Latency 








(Timestamp of Response - Timestamp of Request) 
Round-Trip Time - Time Spent on Server 


花 营 在 服务 器 上 的 时 间 可 以 由 服务 器 来 计算 。 对 客户 端 而 言 ， 往 返 的 时 间 是 准确 可 用 的 。 
服务 器 可 以 将 花费 的 时 间 放 在 响应 的 自 定义 头 部 ， 然 后 客户 端 就 可 以 用 来 计算 延迟 了 。 
例 7-1 给 出 了 计算 延迟 的 示例 代码 。 该 代码 假设 啊 应 包含 了 自 定 义 头 部 X-Server-Time, jX 
个 时 间 以 毫秒 为 单位 ， 包 含 在 服务 器 上 花费 的 时 间 。 


例 7-1 计算 网 络 延迟 


//server - NodeJS 
app.post("/some/path", function(req, res) { 














var startTime - new Date().getTime(); 
// 处 理 
var body = processRequest(req); 


var 
var 
res 
res 


})s 


endTime = new Date().getTime(); 

serverTime = endTime - startTime; 
.header("X-Server-Time", String(serverTime)); 
.send(body) ; 


//client - iOS app 
-(void)fireRequestWithLatency:(NSURLRequest *)request { 


NSDate *startTime = nil; 
AFHTTPRequestOperation *op = 


[op 


[[AFHTTPRequestOperation alloc] initWithRequest: request]; 
setCompletionBlockWithSuccess:^(AFHTTPRequestOperation *op, id res) ( 


NSDate *endTime - [NSDate date]; 
NSTimeInterval roundTrip - [endTime timeIntervalSinceDate:startTime]; 
long roundTripMillis - (long)(roundTrip * 1000); 


NSHTTPURLResponse *res - op.response; 
NSString *serverTime - [res.allHeaderFields objectForKey:Q"X-Server-Time"]; 


long serverTimeMillis - [serverTime longLongValue]; 


long latencyMillis = roundTripMillis - serverTimeMillis; 





) failure:^(AFHTTPRequestOperation *op, NSError *error) { 
// 处 理 错误 。 如 果 需 要 ,向 用 户 展示 错误 
]]; 








startTime = [NSDate date]; 
[op start]; 


j 


151 7-1 中 的 代码 关于 网 络 延 迟 的 表达 基本 准确 ， 但 它 包 括 了 服务 端 线路 上 刷新 数据 花费 的 
时 间 ， 以 及 在 客户 端 解析 响应 花费 的 时 间 。 如 果 可 以 分 离 出 来 ， 它 将 提供 真实 的 网 络 延迟 
时 间 ， 包 括 任何 设备 开销 。 
如 有 果 你 有 数据 来 分 析 任 何 模式 下 的 延迟 ， 还 需 跟踪 下 列 数据 。 
。 连接 超时 
跟踪 连接 超时 的 次 数 是 非常 重要 的 。 根 据 网 络 质量 〈 较 薄弱 的 基础 设施 或 较 低 的 容量 )， 
该 指标 会 提供 详细 的 地 理 区 域 分 类 , 网络 质量 将 反 过 来 帮助 规划 同步 时 间 的 传输 。 例 如 ， 
同步 会 在 短 时 间 间 隔 传 输 ， 比 如 几 分 钟 ， 而 不 用 在 某 一 个 特定 时 间 跨 时 区 同步 。 


。 响应 超时 

捉 连 接 成 功 但 响应 超时 的 数量 。 这 有 助 于 根据 地 理 位 置 和 日 期 、 年 份 的 时 间 来 规划 数 
据 中 心 的 容量 。 

。 BK) 
请 求 以 及 响应 的 大 小 完全 可 以 在 服务 器 端 进行 测量 。 使 用 此 数据 可 以 识别 任何 可 能 降 
低 网 络 操作 速度 的 峰值 ， 并 确定 一 些 可 用 选项 : 通过 选择 合适 的 序列 化 格式 (ISON, 
CSV, Protobuf 等 ) 减少 数据 占 位 ， 或 者 分 割 数 据 并 使 用 增 量 同步 〈 例 如 ， 通 过 使 用 小 
的 批量 大 小 或 在 多 个 块 中 发 送 部 分 数据 ) 。 





































































































较 差 网 络 容量 的 最 大 化 
我 曾经 开发 过 一 个 神奇 的 体育 应 用 ， 当 时 的 工程 师 团队 渐渐 注意 到 应 用 的 延迟 变 得 
更 长 ， 超 时 (连接 和 响应 超时 ) 变 得 更 多 了 。 我 们 还 发 现 ， 服 务 器 通常 会 发 送 超过 
200KB 的 压缩 ISON 数据 作为 初始 开销 ， 因 此 我 们 必须 在 比赛 开始 前 约 20 分 钟 内 做 这 
件 事情 。 
在 比赛 当天 晚上 ， 现 场 有 超过 10 000 个 用 户 连 接 至 一 个 基站 ， 总 共有 50 000-80 000 
个 用 户 ， 造 成 了 移动 数据 网 络 的 阻塞 。 


虽然 不 能 做 任何 事情 改善 连接 ， 但 我 们 使 用 了 一 些 技巧 来 改善 体验 。 开 始 时 ， 我 们 向 
设备 发 送 了 推送 通知 。 在 开始 前 几 个 小 时 发 送出 去 的 第 一 个 推送 通知 用 于 询问 用 户 是 
否 将 要 去 比赛 场 。 并 非 所 有 用 户 都 回应 了 ， 但 相当 多 的 用 户 回应 了 (我 们 采用 了 游戏 
化 来 激励 ) 。 这 不 仅 提 供 了 估算 流量 的 数据 ， 更 重要 的 是 明确 了 哪些 用 户 需要 通知 。 

第 二 个 推送 通知 只 发 送 给 了 表示 将 要 前 往 比 赛场 的 用 户 。 这 个 推送 通知 是 在 比赛 的 前 
20 分 钟 分 批发 送出 去 的 。 如 果 球 场 有 1000 个 用 户 ， 其 中 的 100 个 用 户 会 在 初始 的 两 
分 钟 收 到 通知 ， 下 100 个 用 户 会 在 接 下 来 的 2 分 钟 收 到 通知 ， 并 依次 类 推 。 











fe 
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通知 将 唤醒 应 用 ， 通 过 使 用 地 理 位 置 来 决定 是 否 获取 数据 。 现 在 不 再 是 1000 个 人 同时 
连接 ， 连 接 将 在 以 100 人 为 一 组 的 用 户 组 中 同时 进行 。 

显而易见 ， 你 不 能 指望 每 个 用 户 立 即 打 开 应 用 ， 但 一 个 推送 通知 将 有 助 于 唤醒 应 用 ， 
并 让 它 同步 数据 。 与 应 用 进一步 交互 时 会 变 得 更 加 顺畅 。 











7.1.5 网 络 API 
在 执行 任何 网 络 操作 时 ， 重 要 的 是 你 选择 的 API 


iOS 的 早期 版 本 提供 了 NSURLConnection 来 执行 网 络 请 求 。 应 用 的 开发 人 员 需 要 管理 连接 
池 ， 并 处 理应 用 后 台 、 中 断 以 及 请 求 的 恢复 。 


NSURLSession 于 iOS 7 推出， 现在 应 该 是 执行 任何 网 络 操作 的 实际 选择 。 使 用 NSURLSession 
BALA PUR. | 


e NSURLSession 对 于 放 入 其 中 的 相关 请 求 而 言 是 一 个 可 配置 的 容器 。 例 如 ， 对 服务 器 的 所 
有 请 求 都 可 配置 成 始终 包含 访问 令 牌 。 

。 你 可 以 得 到 后 台 联 网 的 所 有 好 处 。 这 有 助 于 提高 电池 寿命 、 支 持 UIKit 的 多 任务 、 使 用 
与 过 程 传输 相同 的 委托 模型 。 

。 任何 网 络 任务 都 可 以 暂停 、 停 止 并 重新 启动 。 与 NSURLConnection 不 同 ， 此 处 并 不 需要 
是 NSOperation 的 子 类 。 

e 你 可 以 继承 NSURLSession 来 配置 会 话 ， 以 便 在 每 个 会 话 的 基础 上 使 用 专用 存储 (缓存 、 
cookie jar 等 ) 。 

e 当 使 用 NSURLConnection 时 ， 如 果 遇 到 了 身份 验证 问题 ， 问 题 会 在 任何 一 个 请 求 中 返回 ， 
你 无 法 明确 知道 哪个 请 求 遇 到 了 这 个 问题 。 使 用 NSURLSession， 委 托 会 处 理 身份 验证 。 

e NSURLConnection 有 一 些 基 于 块 的 异步 方法 ， 但 委托 不 能 使 用 它们 。 当 发 起 一 个 请 求 时 ， 
要 么 成 功 ， 要 么 失败 ， 即 使 它 需要 身份 验证 。 

e 使 用 NSURLsession， 你 可 以 采取 一 种 混合 的 方法 ， 这 也 就 是 说 ， 你 可 以 使 用 基于 块 的 异 
步 方法 ， 还 可 以 设置 委托 以 处 理 身 份 验证 。 


7.2 ”应 用 部 署 


随 着 对 这 些 指标 的 统计 ， 你 可 以 更 好 地 规划 应 用 的 部 署 。 这 不 仅 包括 服务 器 、 服 务 器 的 位 
置 和 容量 ， 还 包括 客户 端 ， 以 及 如 何在 给 定 的 场景 下 获得 最 好 的 。 


在 本 节 中 ， 我 们 将 从 网 络 的 角度 研究 端 到 端 应 用 的 重要 组 件 一 在 掌控 之 中 的 那些 组 件 。 




































































注 1: 使 用 的 有 关 提 示 请 参见 Ken Toh 的 “NSURLSession Tutorial: Getting Started” (http://www.raywenderlich. 


com/51127/nsurlsession-tutorial ) 。 





7.2.4 服务 器 

在 查看 网 络 延迟 的 地 域 分 布 时 ， 我 们 可 以 使 用 这 个 信息 为 数据 中 心 选择 适当 的 位 置 。 如 果 
使 用 托管 的 数据 中 心 提供 商 ， 不 妨 选 择 有 多 个 地 理 位 置 的 ， 如 Amazon AWS 或 Rackspace 
Cloud。 如 果 你 有 自己 的 数据 中 心 ， 那 么 应 该 确保 它们 在 地 理 上 是 分 散 的 。 

无 需 多 想 ， 服 务 器 应 该 安装 在 多 个 位 置 ， 这 样 你 可 以 更 好 地 服务 本 地 内 容 。 

以 下 是 一 些 应 该 遵循 的 最 佳 实践 。 

。 使 用 多 个 数据 中 心 ， 让 服务 器 在 地 理 上 分 散 开 来 ， 更 贴近 用 户 。 

。 使 用 CDN 提供 静态 内 容 ， 如 图 像 、JavaScript、CSS、 字 体 等 。 

。 使 用 接近 的 边缘 服务 器 (http://serverfault.com/a/67489) 来 提供 动态 内 容 。 

。 避免 使 用 多 个 域名 (DNS 查询 时 间 可 能 会 很 长 ， 这 会 降低 用 户 体验 )。 

注意 ， 第 二 点 和 第 四 点 是 互 扩 的， 你 需要 进行 权衡 。 当 使 用 DNS 时 ， 最 大 限度 减少 DNS 
查找 时 间 的 信息 请 参见 7.1.1 节 中 讨论 的 最 佳 实践 。 




































































7.2.2 请求 

为 了 恰当 地 设置 网 络 ， 正 确 地 配置 HTTP/S 请 求 很 重要 。 你 应 该 遵循 以 下 的 最 佳 实践 。 

。 不 要 为 每 一 个 操作 单元 都 进行 一 次 请 求 ， 使 用 批量 请 求 。 即 使 必须 实现 多 个 后 端子 系统 
来 完成 ， 但 是 合并 批量 请 求 会 带 来 较 大 的 性 能 提升 ， 所 以 还 是 值得 的 。 
客户 端 可 以 向 多 个 后 端 发 送 多 路 复 用 的 请 求 ， 而 服务 器 可 以 使 用 多 部 分 /混合 回复 作为 
回应 。 客 户 端 将 对 回复 进行 复 用 。 图 7-7 展示 了 实现 这 一 功能 的 概要 图 。 


























创建 并 处 理 请 求 





后 端 服务 器 





后 端 服务 器 


边缘 服务 器 发 送 响应 1 
响应 
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7-7: 请 求 的 多 路 复 用 系统 




















使 用 持续 的 HTTP 连接 ， 该 连接 也 被 称 为 HTTP 长 连接 。 它 们 有 助 于 最 大 限度 地 减少 
TCP 和 SSL 握手 的 消耗 ， 同 时 也 减少 了 网 络 拥 塞 。 
或 者 使 用 WebSockets。Square 的 SocketRocket (https://github.com/square/SocketRocket) 
这 样 的 库 可 以 协助 人 们 在 10S 上 使 用 WebSocket。 


在 任何 可 以 的 情况 下 都 使 用 HTTP/2 (https://http2.github.io/)。 通 过 单一 的 连接 ， 
HTTP/2 支持 HTTP 请 求 的 真正 复 用 ， 如 果 请 求解 析 为 一 个 全 地址 ， 那 么 HTTP/2 会 将 
跨越 了 多 个 子 域 的 请 求 聚集 到 一 起 ，HTTP/2 还 支持 报头 压缩 等 。 使 用 HTTP/2 的 好 处 
是 巨大 的 。 最 好 的 是 ， 就 消息 结构 而 言 ， 该 协议 仍旧 保持 不 变 ， 依 然 包 括 头 部 和 主体 。 
使 用 HTTP 缓 存 头 设置 正确 的 缓存 级 别 。 对 于 想 要 下 载 的 标准 图 像 ( 如 主题 背景 或 表情 )， 
内 容 的 有 效 期 可 以 设置 为 较 长 的 时 间 。 这 不 仅 保证 了 网 络 库 在 本 地 缓存 它们 ， 还 保证 
了 其 他 设备 可 以 从 在 本 地 进行 了 缓存 的 中 介 服 务 器 (ISP 服务 器 或 代理 ) 中 受益 。 影 响 
HTTP 缓存 的 响应 头 是 Last-Modified、Expires、ETag 和 Cache-Control。 















































7.2.3 数据 格式 


选择 正确 的 数据 格式 和 选择 网 络 参数 一 样 重要 。 一 些 选 择 可 能 会 使 应 用 的 性 能 产生 很 大 的 
不 同 ， 比 如 对 无 损 图 像 压 缩 使 用 PNG 还 是 WEBP。 


如 果 你 的 应 用 是 以 数据 为 导向 的 ， 那 么 选择 适合 其 传输 的 正确 格式 很 关键 。 其 他 协议 支持 
的 功能 也 可 以 提供 帮助 。 


需要 注意 的 是 ， 虽 然 使 用 了 SSL， 但 也 可 能 会 有 安全 问题 。 






































在 选择 数据 格式 时 ， 你 应 该 遵循 以 下 的 最 佳 实践 。 


使 用 数据 压缩 。 当 传送 JSON 或 XML 这样 的 文本 内 容 时 ， 这 一 点 尤为 重要 。 
NSURLRequest 会 自动 给 头 部 添加 Accept-Encoding: gztip、defLate， 这 样 你 就 无 需 自己 
动手 了 。 但 这 也 意味 着 服务 器 应 该 承认 头 部 ， 并 使 用 适当 的 传输 编码 发 送 数据 。 

选择 正确 的 数据 格式 。 不 用 多 想 ，JSON 和 XML 这 样 元 长 、 人 类 可 读 的 格式 是 资源 密 
集 型 的 一 一 序列 化 、 传 输 、 反 序列 化 会 比 使 用 自 定 义 制 作 的 、 二 进 制 的 、 机 器 友好 的 格 
式 更 耗费 时 间 。 此 处 不 讨论 媒体 压缩 ( 即 图 像 压 缩 和 视频 编 解码 器 )， 而 是 着 上 腿 于 文本 
数据 格式 。 

原生 应 用 最 常 选用 的 数据 格式 正好 是 ISON 和 XML。 唯一 的 原因 是 ，web 服务 /API 是 
为 Web 编写 的 ， 并 且 用 于 移动 端 。 

但 是 ， 如 果 你 还 没有 准备 好 ， 那 就 需要 开始 思考 移动 端 了 。 前 面 提 到 的 格式 是 很 方便 手 
工 制作 的 ， 但 对 机 器 操作 而 言 却 是 资源 密集 型 的 。 最 好 从 大 小 以 及 序列 化 / 反 序 列 化 的 
角度 综合 来 看 ， 选 择 一 个 更 优 的 格式 。 


对 于 运输 记录 而 言 ， 最 流行 的 二 进 制 格 式 是 Protocol Buffers， 也 被 人 称 之 为 Protobuf。 
其 他 协议 包括 Apache Thrift 和 Apache Avro。 通 常情 况 下 ，Protobuf 被 认为 要 胜 于 其 他 ， 
但 很 多 东西 还 是 取决 于 使 用 的 数据 类 型 。 如 果 大 部 分 数据 都 是 字符 串 ， 那 么 你 应 该 找 方 
法 优化 它们 的 加 载 ， 因 为 它们 不 会 通过 任何 格式 被 压缩 。 使 用 deflate, gzip 或 任何 无 
损 压 缩 算法 进行 压缩 。 
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7.3.1 网 络 链接 调节 器 


网 络 链接 调节 器 在 IOS 设备 以 及 模拟 器 中 都 是 可 用 的 。 在 设置 应 用 中 ， 你 可 以 通过 开发 者 
图 7-8 显示 了 如 何 获取 网 络 链接 调节 器 和 相关 设置 。 





菜单 进行 访问 。 
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你 可 以 选择 一 个 预定 义 的 配置 文件 ， 或 创建 一 个 新 的 配置 文件 。 通 过 控制 重要 的 参数 ， 网 
络 链接 调节 器 可 以 模拟 不 同 的 网 络 条 件 : 
。 入 站 通信 

带宽 、 数 据 包 丢失 和 延迟 〈 响 应 延迟 ) 
站 通信 


出 
人 带宽、 数据 包 丢 失 和 延迟 














查找 延迟 
。 BMXIPv4, IPv6, 或 两 者 
。 界面 
WiFi、 移 动 ， 或 两 者 
你 应 该 使 用 网 络 链接 调节 器 来 测试 应 用 在 极端 情况 中 的 行为 。 你 可 能 不 会 定期 做 这 些 测 
et 次 。 


7.3.2 AT&T 应 用 资源 优化 器 


虽然 官方 文档 (http://developer.att.com/application-resource-optimizer/docs) 指出 ，AT & T 
的 应 用 资源 优化 器 (http://developer.att.com/campaigns/application-resource-optimizer) 工具 
可 以 优化 移动 web 应 用 的 性 能 ， 但 它 也 可 用 于 本 地 应 用 。 


要 想 使 用 此 工具 ， 需 要 将 iPhone/iPad 配置 为 开发 和 调试 状态 (使 用 Xcode)， 这 样 你 就 可 
以 看 到 前 面 讨 论 过 的 开发 者 菜单 了 ， 同 时 你 还 需要 Mac 上 的 管理 权限 。 

你 可 以 通过 以 下 步骤 启用 开发 者 菜单 。 
(1) 将 iOS 设备 连接 到 Mac OS X 设备 。 

(2) 打开 Xcode。 

(3) 导航 到 Window — Devices., 

(4) 选择 iOS 设备 并 选择 “用 于 开发 ”。 

应 用 资源 优化 器 工具 包括 两 个 步 又: 数据 收集 和 数据 分 析 。 为 了 收集 数据 ， 请 按照 下 列 步 
又 操作 。 

(1) 导航 到 Menu Bar 一 Data Collector 一 Start Collector, 

(2) 在 设备 上 运行 应 用 。 

(3) 导航 到 Menu Bar — Data Collector — Start Collector, 


通过 对 比 一 系列 的 最 佳 实践 测试 ， 数 据 分 析 器 会 评估 从 应 用 中 收集 到 的 数据 。 结 果 在 总 结 
屏幕 中 展示 了 出 来 (如 图 7-9 所 示 )。 
















































































ese AT&T Application Resource Optimizer (ARO) - /Users/: "AROTracelOS/hpert 
File Profile Tools View Data Collector DefectTracking Help | 











CELERE Overview | Diagnostics | Statistics | Waterfall | 








FILE DOWNLOAD 





Date: Mar 2, 2015 11:36:57 PM AT&T Application Resource Optimizer 
Trace: hperf 

Application(s) Name ; Version: Unknown App 

Data Collector Version: 4.0.0 

Device Make/Model: Apple / iPhone5,1 

OS/Platform Version: 8.1.3 

Network Type(s): NONE 

Profile: AT&T 3G 


Reducing the usage of network for file downloads can reduce your application's battery consumption. 


© Test: Text File Compression 
About: Sending compressed files over the network will speed delivery, and unzipping files on a device is a very low overhead operation. Ensure that all your 
text files are compressed while being sent over the network. Learn more... 


Results: AT&T ARO detected 0 text files above 850 bytes were sent without compression. Adding compression will speed the delivery of your content to your 
customers. (Note: Only files larger than 850 bytes are flagged.) 


© Test: Duplicate Content 
f About: This test measures duplicate content. Excess duplicate content means that content was downloaded multiple times, which leads to slower 
applications and wasted bandwidth. Learn more... 





Results: Your trace passes with an acceptable level of duplicate content. Your trace had less than 3 duplicate items downloaded. 


© Test: Cache Control 


About: This test measures the presence of cache headers. For all content that should be stored in the cache the best practice is to make sure that your 
server is adding the appropriate cache headers. Learn more... 


Results: Your data is populated with cache headers and it passes this test. 


























图 7-0: 应 用 资源 优化 器 数据 分 析 器 : 总 结 


该 工具 在 -/AROTracelOS/( name )/ 中 收集 数据 。 在 收集 的 总 体 数 据 中 ， 以 下 是 比较 重要 的 : 


。 设备 的 详细 信息 (型 号 、 操 作 系统 版 本 ， 屏 幕 尺寸 等 ) 
。 通过 pcap 接口 的 流量 的 详细 信息 

。 电池 消耗 

图 7-10 显示 了 应 用 资源 优化 器 工具 中 的 网 络 分 析 汇总 图 。 





Trace: hperf Total Bytes: 6492993 Profile: 





Throughput 


Packets UL 























图 7-10. 应 用 资源 优化 器 数据 分 析 器 : 网 络 用 量 的 诊断 


这 种 分 析 以 一 种 易于 理解 的 方式 呈现 出 来 ， 但 可 以 通过 编程 的 方式 分 析 原 始 数 据 ， 从 而 获 
得 更 详细 的 报告 ， 更 重要 的 是 ， 历 史 数 据 可 用 来 分 析 以 时 间 为 轴线 的 性 能 变化 。 
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7.3.3 Charles 


Charles (http://www.charlesproxy.com) 


配置 ， 以 完成 以 下 操作 。 
。 监视 HTTP 请 求 


EH 


可 以 监视 HTTP 流量 ， 包 括 请 求 、 响 应 数据 以 及 HTTP k, 





应 用 的 示例 请 求 。 





图 7-12 显示 了 同一 请 求 的 响应 。 响 应 是 ISON 格式 的 ， 可 以 在 


任何 外 部 工具 的 情况 下 对 其 进行 格式 化 。 


是 一 个 非常 强大 的 网 络 调 试 代理 。 你 可 以 对 其 进行 





图 7-11 显示 了 来 自 Facebook 
不 使 用 
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Charles 3.9.3 - Session 1 * 


0997/7 X5 





E] Sequence 





h ubersearch.get 
$ logging.clientevent 
») legging.clientevent 
(^ legging.clientevent 
1 logging.clientevent 
Th ubersearch.get 
; ubersearch.get 
ubersearch.get 
| ubersearch.get 
$ ubersearch.get 
Th ubersearch.get 
logging.clientevent 
(^ logging.clientevent 
ubersearch.get 
h ubersearch.get. 
I ubersearch.get 
5 logging.clientevent 
0) ubersearch.get 
3% «unknown» 
其 «unknown» 
3€ «unknown» 
3€ <unknown> 
3€ <unknown> 
> @ https://m.facebook.com:443 
> @ https;//focdn-dragon-a.akar 
上 @ https: //focdn-photos-b-a.aki 
> @ https://fbexternal-a.akamaihr 














Chart Notes | 








Overview [BEES] Response | Summary 
query dan 
uuid 7-4ac1-8a70-6: 
context mobile search. 
photo size 96 
cached ids 0 
filter. ['shortcut,, ‘app’, ‘user’, ‘page’, ‘group’, 'event] 
support groups icons — true 
'group. icon scale 2 
include, is verified true 
mo profile image urls false 
limit. 10 
include, native android url true 
format json 
locale en_US 
client, country, code us 
method ubersearch.get 


fo_api_req_friendly name fetch uberbar result 
fb api caller class com.facebook.search.api. 











POST. https://apl facebook.com/method/ubersearch.get.. 
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ETM sea 
了 @ https://api.facebook.com 
Y liar method/ 
Th ubersearch.get 
ih logging.clientevent 
i logging. clientevent 
让 logging.clientevent 
©) logging.clientevent 
Th ubersearch.get 
()) ubersearch.get 
Th ubersearch.get 
D ubersearch.get 
D ubersearch.get 
Th ubersearch.get 
Th logging.clientevent 
Th logging.clientevent 
Th ubersearch.get 
(| ubersearch.get 


uence | 





(©) logging.clientevent 
() ubersearch.get 

其 <unknown> 

9% «unknown» 

3€ <unknown> 

3€ «unknown» 

3€ <unknown> 
> © https://m.facebook.com:443 
> @ https://focdn-dragon-a.akarr 
> @ https://fbcdn-photos-b-a.ak: 
> @ nttps://foexternal-a.akamaih 








Overview 


Request Summary 


Chart | Notes | 








“subtext” 
“photo” 
sGategory"s "Artist at 
"is verified": false, 
"friendship status”: 
pt 

wid": 74", 

type": "user", 

npath"s "httparV/V/wew- facebook. comi, i 

"(Prince Dante)", 


ion \u00b7 ViotDreamorz 





can request" 









false, 
Tendship status": "cannot request 
"vid": 
"type": 
"path": 
"text: 
"subtext' 
photo": 
"category 


1000; , 

user 

https: \/\/wew. facebook.com\/nature. j 
ature-in Danger", 









ature (journal)", 


“is verified": false, 
"friendship status": 


ht 
"vid" 


"can request" 


"100005. , 

user 
https: \/\/www. facebook. com /danie^ 
"De Medeiros", 





MM £bcdn-profile- 
£ false, 
friendship status": “ci 








request“ 





1000044:... ^" 
“user”, 
hittoa) RE 





“https:\/\/fbedn-profile-a.akamaihd.net\/hprofile-ak-xfp1\/v\/t1.0-1\/p100x100\/ 


\/\/£bedn-profile-a.akamaihd.net\/hprofile-ak-xfal\/v\/t1.0-1\/c13.0.100. 


netps:\/\/fbodn-profile-a-akamaihd.net\/hprofile-ak-xapl\/v\/t1.0-1\/626.8.96.9 


-net\ /hprofile-ak-x£a1\/v\/t1.0-1\/c26.8.96.9 





Headers Text Hex Compressed 


ox BESTEN aw) 








POST htios://apl facebook.com/method /ubersearch.oet 
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监控 HTTPS 请 求 
这 需要 设置 证 书 。 





你 可 以 创建 


charlesproxy.com/charles.crt ) , 


请 求 了 。 





如 








pa 








自己 的 签名 证 书 或 使 用 网 站 提供 的 默认 证 书 (http:/ 


7-13 所 示 。 在 设备 上 安装 证 书 ， 就 可 以 查看 HTTPS 





Throttling 
Throttle Settings... 
Breakpoints... 

Reverse Proxies... 
Port Forwarding... 


Mac OS X Proxy 





Tools Window Help 
V Recording (Session 1) 
Recording Settings... 


8R 


OST 
ORK 


HO | 
MP | 


Mozilla Firefox Proxy 88F ， 


| 
Access Control Settings... 
External Proxy Settings... 1 
Web Interface Settings... 
Client SSL Certificates... k 


Oni gau 1 


ocations 


aa 


&(€(&& € & & (& (& 


Proxy Settings 





| Proxies — Options MacOSX Mozilla Firefox | 





Charles can show you the plain text contents of SSL requests and responses. Only the locations listed 
below will be proxied. Charles will issue and sign SSL certificates, please press the Help button for 
more information. 


xr] M Enable SSL proxying 


_| Use a custom CA certificate 
Choose 





m443 
2m:443 
com:443 
/Lcom:443 
».com.443 
om:443 
^^m443 
2.com:443 
...Comi443 
.com;443 
ema 


Add | Remove 


[ Cancel | 
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。 发 送 自 定义 响应 


这 是 一 个 非常 有 用 的 功能 ， 可 以 在 不 打扰 生产 服务 
测试 性 能 ， 就 发 送 大 量 的 数据 。 要 测试 稳定 
你 可 以 进入 Tools 一 Rewrite 一 Enable Rewrite 一 Add， 如 


置 响应 ， 如 图 7-15 所 示 。 





仅 在 测试 设备 上 使 用 网 站 的 密 钥 。 私 钥 和 公 钥 都 在 
备 上 使 用 它们 ， 你 将 会 面临 很 大 的 风险 一 一 所 有 的 3 


共 领 域 。 
都 可 能 被 监控 。 











如 果 在 个 人 设 


器 的 前 提 下 测试 各 种 可 能 的 情况 。 要 
性 ， 就 发 送 大 量 的 数据 以 及 无 效 的 输入 。 
图 7-14 所 示 ， 并 根据 URL 配 
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ece Rewrite Settings 














Modify requests and Name: | HPerf Photos 
responses as they pass EON 
Window Help through Charles. Sts 
No Caching... BN @ Enable Rewrite | M. https: //hperf.appspot.com/v1/feed 
Block Cookies... TC (C Debug in Error Log 
Map Remote... TEM WEE | 


Map Local... TEL 
Black List... TB 
DNS Spoofing... ED 


| 
| 


HPerf Photos 





[Remove 


Mirror... RI 
Auto Save... THA 


Client Process... Rules 





TER 





| Æ Body {'status": 200, "photos": (I) 





















































| 
Publish Gist | 
Publish Gist Settings... | 
Import/Export Settings... | 
| Add JI Remove || Up || Down | 
© [_Cancei_) ok {Apply | 
& 7-14: Charles: 开启 自 定义 响应 
e^o Rewrite Rule 
Type: | Body > 
e Where 
[VÍ Request C Response 
, Match 





Enter text to match or leave blank to match all. 
Name: |] Regex 


Value: | | [L] Regex 


M Match whole value |.) Case sensitive 


, Replace 一 


Name: 





Value: | {"status": 200, "photos": []) 


( ) Replace First (*) Replace All 


Enter new values or leave blank for no change. If using regex 
matches you may enter references to groups, eg. $1 











© C cance] [7797] 














& 7-15; Charles; 设置 自 定义 响应 
从 开发 、 调 试 和 测试 的 角度 来 看 ， 请 记 住 ， 会 话 日 志 可 以 被 保存 并 分 发 给 团队 进行 分 析 。 





7.4 小 结 


了 解 与 网 络 相关 的 重要 指标 以 及 用 来 测量 网 络 的 相关 技术 ， 将 帮助 你 开发 高 性 能 的 应 用 ， 
让 应 用 可 以 最 大 限度 地 减少 任何 资源 的 利用 率 ， 从 而 获得 更 好 的 用 户 体验 。 

这 些 做 法 通常 需要 进行 权衡 ， 因 为 对 多 个 参数 进行 优化 时 ， 所 需 的 条 件 可 能 会 产生 竞争 关 
系 ， 所 以 需要 取得 平衡 。 例 如 ， 为 安全 性 使 用 SSL 与 内 存 利 用 率 和 执行 速度 的 权衡 ， 为 了 
提高 速度 使 用 可 缓存 静态 内 容 的 CDN 与 为 速度 使 用 单个 主机 名 的 权衡 ， 等 等 。 

不 要 依赖 网 络 状 态 ， 因 为 它 随时 都 可 以 更 改 。 你 最 好 能 确保 自己 的 网 络 层 适应 于 网 络 类 型 
和 状态 。 具 体 来 说 ， 对 于 媒体 流 ， 使 用 自 适 应 多 比特 率 HLS。 对 于 非 媒 体内 容 ， 执 行 批 处 
时 操作 ， 并 尝试 尽 可 能 多 地 预先 下 载 。 请 务必 提供 适合 设备 的 数据 ， 因 此 需要 考虑 设备 的 
性 能 、 大 小 和 外 形 尺寸 。 
工具 应 该 提供 不 同 的 网 络 条 件 以 及 不 同 的 数据 响应 来 帮助 你 测试 应 用 。 使 用 这 些 工 具 来 测 
试 与 网 络 、 请 求 和 响应 相关 的 方方面面 ， 从 而 加 固 应 用 。 
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有 时 你 会 需要 与 其 他 应 用 共享 数据 ， 或 访问 设备 上 其 他 应 用 的 共享 数据 。 共 享 数据 的 场景 

包括 以 下 几 个 。 

。 与 其 他 应 用 集成 〈 例 如 ， 让 用 户 使 用 Facebook 的 登录 信息 登录 你 的 应 用 )。 

。 发 布 一 系列 互补 的 应 用 ， 比 如 ，Google 提供 的 应 用 〈 如 Gmail, Google Calendar, 
Google Hangouts 和 Google+)。 

。 将 用 户 数 据 从 统一 的 应 用 移动 到 有 多 个 特定 用 途 的 应 用 ， 检 测 其 是 否 存在 ， 并 在 需要 时 
传递 控制 (例如 ，Facebook 应 用 分 为 Messenger、Pages 和 Groups 应 用 ,分 别 用 于 消息 
传递 、 页 面 管 理 和 组 管理 )。 

。 在 可 用 的 最 佳 查 看 器 中 打开 文档 (例如 ， 在 本 地 查看 器 中 查看 PDF 文件 ， 在 Photoshop 
Express 中 编辑 照片 ) 。 


用 于 共享 数据 的 每 种 技术 对 可 共享 的 数据 具有 特定 的 限制 。 例 如 ， 使 用 剪贴 板 会 消耗 大 量 
的 RAM， 文 档 的 共享 会 使 用 设备 存储 (RAM 和 设备 存储 必须 在 使 用 后 清除 )。 类 似 地 ， 
使 用 深层 链接 会 有 数据 序列 化 和 解析 的 开销 。 

在 本 章 中 ， 我 们 将 从 性 能 角度 讨论 各 种 数据 共享 的 选项 ， 并 明确 一 些 特 定 情况 下 的 最 佳 
实践 。 


8.1 深层 链接 


在 移动 应 用 的 上 下 文中 ,深层 链接 包括 使 用 统一 的 资源 标识 符 (uniform resourceidentifier, 
URI) ， 其 链接 到 移动 应 用 内 的 特定 位 置 ， 而 不 是 简单 地 启动 应 用 。 

深层 链接 为 应 用 之 间 的 共享 数据 提供 了 解 耦 的 方案 。 与 访问 网 站 时 的 HTTP 网 址 类 似 ， 
iOS 中 的 深层 链接 通过 所 谓 的 自 定 义 URL scheme 来 提供 。 你 可 以 配置 自己 的 应 用 ， 让 它 
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响应 唯一 的 sheme， 操 作 系统 会 确保 无 论 何 时 使 用 该 sheme， 都 由 你 的 应 用 进行 处 理 。 























应 用 可 以 响应 任何 数量 的 scheme, 
应 用 不 能 响应 以 下 URL scheme 的 保留 列表 。 








http, https 

浏览 网 络 的 标准 scheme; 由 Safari 处 理 。YouTube 链接 是 一 个 例外 ， 如 果 已 安装 应 用 ， 
则 由 YouTube 打开 (这 是 因为 在 创建 自己 的 视频 播放 器 之 前 ， 苹 果 已 经 与 谷歌 形成 了 
合作 伙伴 关系 )。 

mailto 

发 送 电 子 邮 件 的 scheme; 由 邮件 应 用 处 理 ， 如 mailto:email@domain.com, 

itms, itms-apps 

用 于 将 用 户 带 入 应 用 安装 界面 ， 由 App Store 应 用 处 理 。 这 曾经 是 唯一 可 用 的 选项 ， 直 
到 iOS63I 入 T Store Kit (https://developer.apple.com/library/prerelease/ios/documentation/ 
StoreKit/Reference/StoreKit. Collection/index.html) , 







































































tel 
用 于 呼叫 电话 号 码 ， 由 电话 应 用 处 理 ， 如 tel: //1234567890, 


app-settings 
iOS 8 的 新 功能 ， 此 scheme 带领 用 户 去 设置 应 用 ， 并 直接 进入 应 用 的 设置 部 分 。 















































你 选择 的 URL scheme 在 所 有 已 安装 的 应 用 中 必须 是 唯一 的 ， 否 则 会 发 生 未 定义 的 行为 。， 
你 可 以 使 用 以 下 一 种 或 多 种 方法 来 创建 唯一 的 scheme。 


无 论 使 用 何 种 选项 ， 你 唯一 能 期 待 的 就 是 其 他 的 应 用 不 会 使 用 相同 的 Scheme， 因 为 这 
也 许 检 测 不 出 来 。 任 何其 他 的 恶作剧 应 用 如 果 使 用 了 相同 的 Scheme， 而 且 未 被 发 现 ， 
那么 它 可 能 会 继续 拦 蕉 本 应 属于 你 的 应 用 的 链接 。 


选择 唯一 的 scheme 


反 向 的 DNS 符号 

例如 ， 如 果 你 拥有 的 域名 是 yourdomain.com， 那 就 使 用 com. yourdomain.appname, 

& ID 

由 于 所 有 提交 到 App Store 的 应 用 软件 包 的 ID 必须 是 唯一 的 ， 因 此 你 可 以 使 用 此 ID, 


应 用 ID 加 前 级 

在 App Store 中 ， 每 个 应 用 都 有 一 个 唯一 的 数字 ID。 你 可 以 在 它 前 面 加 几 个 字符 ， 
然后 得 到 一 个 唯一 的 ID。 例 如 ， 如 果 应 用 的 ID 是 12345678906， 你 可 以 选择 scheme 
为 10s1234567890 或 app1234567890, 








注 1: 


在 Android 系统 上 ， 如 果 多 个 应 用 响应 一 个 scheme， 那 操作 系统 会 提示 用 户 选 择 一 个 。 类 似 功能 将 
来 有 望 引 入 iOS 中 。 
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除非 选择 一 个 通过 争论 可 以 获得 所 属 权 的 Scheme， 否则 很 难 向 苹果 支持 部 门 提 出 投 
诉 。 所 以 ， 要 选择 一 个 可 以 辩护 的 scheme, 例如 ,使 用 com.yourdomain.appname 这 样 
的 scheme Ft mail 或 song 更 有 说 服 力 。 





iOS 9 引入 了 通用 链接 ， 人 允许 应 用 处 理 http 或 https， 同 时 可 以 验证 域名 的 
所 有 权 。 我 们 将 在 13.1.1 节 中 对 此 进行 讨论 。 











驱动 深层 链接 有 三 个 步骤 。 

(1) 检 测 scheme 是 否 可 以 被 处 理 。-[UIApplication canOpenURL:] 方法 允许 你 在 设备 上 安装 
的 应 用 中 检查 是 否 至 少 有 一 个 应 用 可 以 处 理 特定 的 sctheme。 选 择 唯一 的 scheme 有 助 于 
检测 是 否 安装 了 特定 的 应 用 。 


由 于 scheme 的 唯一 性 ， 自 定义 URL scheme 可 用 于 检测 应 用 是 否 被 安装 。 你 
可 以 使 用 版 本 后 缀 来 检测 是 否 安装 了 特定 版 本 。 

因此 ， 应 用 可 以 支持 多 种 scheme。 例 如 ，Yelp (https:/www.yelp.com/developers/ 
documentation/v2/iphone) 使 用 了 三 个 scheme yelp5.3, yelp4 和 yelp, 4> 
别 对 应 应 用 的 5.3.0 或 更 高 版 本 、4.0.0 或 更 高 版 本 ， 以 及 2.0.0 或 更 高 版 本 。 
因此 ， 如 果 can0penURL: 为 yeLp4 返回 YES， 则 表明 设备 上 安装 了 4.0.0 或 更 
高 版 本 ， 这 可 以 帮助 你 选择 不 同 的 URL， 以 获得 更 好 的 用 户 体验 。 

同样 ， 你 可 以 选择 使 用 com.yourdmain.appname+v1 和 com. yourdmain. 
appname+v2， 分 别 对 应 应 用 的 版 本 1 和 版 本 2。 





















































(2) 通过 URL 开启 应 用 。 一 旦 检测 到 应 用 的 存在 ， 下 一 步 就 是 创建 最 终 的 URL， 并 打开 应 
用 ,使 用 -[UIApplication openURL] 方法 启动 应 用 。 





URL 格式 
URL 的 格式 是 没有 标准 的 ， 其 至 连 约定 也 没有 。 应 用 使 用 了 各 种 样式 。 URL 的 一 般 
格式 是 scheme://host/path?query#fragment， 其 中 的 path T VA4& | iE £F 4x Acme dx 
路 径 。 
以 下 为 使 用 的 一 些 样 式 。 


。 仅 有 路 径 的 URL 
这 个 想法 是 只 使 用 路 径 来 表示 在 处 理 数 据 时 所 需 的 所 有 细节 。 例 如 ，fb://profile/ 
{id}, Facebook 用 它 来 显示 用 户 的 配置 文件 。 
它 具 有 简单 且 易 于 理解 的 优点 。 
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e 基于 路 径 和 查询 的 URL 
这 是 更 为 通用 和 广泛 使 用 的 格式 。 例 如 ，yelp:///search?term=burritos，Yelp 用 
它 来 搜索 特定 术语 。 
x-callback-url (http://x-callback-url.com/) 提出 了 使 用 URL 格式 {scheme}: //{host}/ 
{action}?{x-callback parame ters}&{action parameters) 的 标准 ， 在 该 标准 中 ， 主 机 
总 是 x-callback-url, Tumblr, Google Maps, Google Chrome 和 一 些 其 他 的 应 用 
(http://x-callback-url.com/apps/) 都 支持 此 格式 。 


它 的 优点 是 创建 和 解析 都 较为 简单 。REFC 6874 (https://tools.ietf.org/html/rfc6874) , 
RFC 3986 (https://tools.ietf.org/html/rfc3986) 和 RFC 1738 (https://tools.ietf.org/html/ 
rfcl738) 这 样 的 标准 存在 广义 的 URL 格式 。 标 准 解析 器 的 职责 是 解析 更 复杂 的 、 
可 能 有 转 义 序列 的 查询 字符 串 。 











(3) 在 目标 应 用 中 处 理 链接 。 当 应 用 接收 到 URL 时 ，UIAppDelegate 通过 回调 方法 -[UIApp- 
Delegate application:openURL:sourceApplication:annotation:] 获取 通知 。 解 析 传 入 
的 URL、 提 取 参 数 / 值 、 处 理 ， 然 后 继续 。 


响应 深层 链接 可 能 会 将 用 户 带 往 应 用 的 其 他 部 分 ， 因 此 ， 你 应 该 添加 一 个 选 
项 ， 让 用 户 能 够 返回 应 用 的 前 一 个 部 分 。 一 个 较 好 的 实现 方案 是 使 用 有 限 的 











例 8-1 显示 了 深层 链接 的 示例 一 一 源 和 目标 。 


例 8-1 深层 链接 
// 源 应 用 一 一 某 个 视图 控制 器 
-(void)openTargetApp { 
NSURL *url = [NSURL URLWithString: 
Q"com. yourdomain.app://x-callback-url/quote?ticker-GOOG| 
&start-2014-01-01&end-2014-12-31"]; [1) 
UIApplication *app = [UIApplication sharedApplication]; 
if([app canOpenURL:url]) ( @ 
[app openURL:url]; © 














j 
// 和 否则 显示 错误 
} 
// 目 标 应 用 一 一 应 用 委托 








-(BOOL)application:(UIApplication *)application 
openURL: (NSURL *)url 
sourceApplication:(NSString *)sourceApplication 
annotation:(id)annotation ( @ 


NSString *host = url.host; @ 

NSString *path = url.path; @ 

NSDictionary *params = [self parseQuery:url.query]; © 
if([Q"x-callback-url" isEqualToString:host]) { © 
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if([@"quote" isEqualToString:path]) { © 
[self processQuoteUsingParameters:params]; @ 
} 
} 


return YES; 


-(NSDictionary *)parseQuery:(NSString *)query { 


NSMutableDictionary *dict = [NSMutableDictionary new]; 
if(query) { 
LEK & 和 "= ' 作 为 分 隔 符 解析 
NSArray *pairs = [query componentsSeparatedByString:Q"&"]; G 


for(NSString *pair in pairs) { 
NSArray *kv = [pair componentsSeparatedByString:@"="]; 
NSString *key = [kv.firstObject stringByRemovingPercentEncoding]; 
NSString *value = [kv.lastObject stringByRemovingPercentEncoding]; 


j 


[dict setObject:value forKey:key]; 
} 


return [NSDictionary dictionaryWithDictionary:dict]; 


-(void)processQuoteUsingParameters:(NSDictionary *)params { 


} 


NSString *ticker = [dict objectForKey:@"ticker"]; 
NSDate *startDate = [dict objectForKey:@"start"]; 
NSDate *endDate = [dict objectForKey:@"end"]; © 
// 验 证 并 处 理 





@ 构造 URL. 

O 检查 应 用 是 否 被 安装 。 

© 启动 目标 应 用 。 

Q 接收 URL 的 目标 应 用 中 的 委托 回调 。 

© 从 URL 中 提取 必要 的 详细 信息 ， 包 括 主机 、 路 径 和 查询 。 

















Q 处 理 URL 一 一 此 例 为 检查 主机 和 路 径 。 
O 在 此 示例 中 ， 处 理 引 号 。 
© 解析 查询 字符 串 。 此 代码 未 优化 ， 因 为 它 会 对 字符 串 进 行 两 次 解析 。 























© 处 理 提取 的 值 。 不 要 忘记 验证 。 


不 论 是 访问 共享 数据 ， 还 是 对 外 共享 数据 ， 深 层 链接 可 能 是 最 常用 的 选项 ， 同 时 ， 优 化 创 
建 和 解析 的 时 间 也 很 重要 。 以 下 列表 涵盖 了 可 以 遵循 的 一 些 最 佳 实践 ， 从 而 让 应 用 实现 最 
优 性 能 。 








。 最 好 使 用 较 短 的 URL， 因 为 它们 的 构建 速度 和 解析 速度 都 比较 快 。 
。 避免 基于 正则 表达 式 的 模式 。 


如 果 使 用 Button Deep Link SDK (http:Wwww.usebutton.com/sdkydeep-links/) ， 它 将 使 用 基 
于 路 径 的 URL 和 基于 此 的 正则 表达 式 。 例 如 ， 路 径 模式 {scheme}//say/:title/:message 
(https://github.com/button/DeepLinkKit/blob/master/SampleApps/ReceiverDemo/ 





DPLReceiverAppDelegate.m) 需要 两 个 正则 表达 式 一 一 一 个 用 于 和 斜 线 分 隔 符 ， 一 个 用 于 
提取 参数 名 称 。 

。 优先 选择 基于 查询 的 URL 进行 标准 解析 。 用 基于 字符 的 分 隔 符 解析 比 使 用 正则 表达 式 
解析 更 快 。 

。 在 你 的 URL 中 支持 深层 链接 回调 ， 以 帮助 用 户 完成 意图 。 一 个 较 好 的 方法 是 支持 三 个 
选项 : success, failure 和 cancel。 
例如 ， 如 果 一 个 照片 编辑 器 应 用 能 让 用 户 将 编辑 的 照片 返回 至 照片 应 用 中 ， 那 么 必然 会 
很 受 欢 迎 。 另 一 个 示例 是 ， 如 果 应 用 用 于 验证 身份 ， 那 么 提供 一 个 能 将 用 户 带 回 到 源 应 
用 的 选项 ， 并 带 上 登录 是 否 成 功 、 是 否 被 取消 或 失败 的 详细 信息 。 
x-callback-url 规范 支持 这 些 回调 。 

* URL 最 好 使 用 深层 链接 ， 以 帮助 用 户 定义 一 个 需要 多 个 应 用 协调 的 工作 流 。 
例如 ， 用 户 可 能 想 要 完成 以 下 操作 。 
(1) 拍摄 照片 。 
(2) 编辑 照片 。 
(3) 将 更 新 后 的 照片 发 送 给 家 人 和 朋友 。 
(4) 在 社交 媒体 上 分 享 更 新 后 的 照片 。 
(5) 最后， 返回 照片 应 用 来 拍摄 下 一 张 照 片 。 


第 一 个 应 用 可 以 定义 深层 链接 的 目标 应 用 列表 ， 以 及 在 完成 整个 过 程 后 所 调用 的 最 终 的 
深层 链接 。 


。 不 要 在 URL 中 放置 任何 敏感 数据 。 有 具体 来 说 ， 不 要 使 用 任何 身份 验证 令 牌 。 这 些 令 牌 
可 能 会 被 未 知 的 应 用 劫持 。 

。 不 要 信任 任何 传 入 的 数据 。 始 终 验 证 URL。 作 为 附加 的 措施 ， 可 以 让 应 用 在 传递 URL 
前 对 数据 进行 签名 ， 并 在 处 理 前 验证 签名 ， 这 可 能 会 是 个 不 错 的 主意 。 但 是 ， 为 了 安全 
地 进行 ， 私 钥 必 须 保存 在 服务 器 上 ， 如 此 一 来 ， 就 必须 要 有 网 络 连 接 。 

e 使 用 sourceApplication 来 标识 源 。 有 一 个 应 用 白 名 单 非常 有 用 ， 你 可 以 始终 信任 这 些 数 
据 。sourceApplication 的 使 用 与 签名 验证 不 正 交 。 这 可 以 是 URL 开始 处 理 前 的 第 一 步 。 


8.2 剪贴 板 


官方 文档 对 剪贴 板 的 描述 如 下 。 


"E c eu erg cd 许多 操作 取决 
于 剪贴 板 ， 特 别 是 复制 一 AAA ee 但 你 也 可 以 在 其 他 情况 下 使 用 剪贴 
板 ， 例 如 ， 


可 通过 UIPasteBoard 类 使 用 剪贴 板 ， 该 类 可 以 访问 共享 存储 库 ， 写 对 象 和 读 对 象 在 共享 存 
储 库 中 进行 数据 交换 。 写 对 象 也 被 称 为 剪贴 板 所 有 者 ， 将 数据 存储 在 剪贴 板 实例 上 。 读 对 




































































iE 2: iOS Developer Library, “Pasteboard” (https://developer.apple.com/library/ios/documentation/General/Conceptual/ 
Devpedia-CocoaApp/Pasteboard.html). 

















象 访 问 剪 贴 板 ， 将 数据 复制 到 其 地 址 空间 中 。 

剪贴 板 可 以 是 公共 的 或 私有 的 。 每 个 剪贴 板 必 须 有 唯一 的 名 称 。 

剪贴 板 可 以 保存 一 个 或 多 个 实体 ， 这 些 实体 被 称 为 剪贴 板 项 目 。 每 个 项 目 可 以 有 多 个 表述 。 
图 8-1 显示 了 一 个 具有 两 个 项 目的 剪贴 板 ， 每 个 项 目 都 有 多 种 格式 。 


。 具有 两 种 标准 格式 (RTF 和 纯 文 本 ) 内 容 的 文本 项 。 
。 具有 两 种 标准 格式 (JPG 和 PNG) 和 一 种 私有 格式 (com.yourdomain.app.type) 图 片 内 
容 的 图 片 项 目 ， 为 特定 应 用 所 专用 。 
































应 用 粘贴 板 (2 个 项 目 ) 














8-1; 剪贴 板 项 目 表述 
例 8-2 显示 了 共享 数据 的 示例 代码 。 


例 8-2 使 用 剪贴 板 共 享 数据 


// 共 享 至 公共 的 剪贴 板 
-(void)shareToPublicRTFData:(NSData *)rtfData text:(NSString *)text { 
[[UIPasteboard generalPasteboard] setData:rtfData forPasteboardType:kUTTypeRTF]; @ 




















[[UIPasteboard generalPasteboard] setData:text forPasteboardType:kUTTypePlainText]; 
[UIPasteboard generalPasteboard].string - text; 
[UIPasteboard generalPasteboard].strings = Q[text]; © 

} 


// 假 设 数据 的 UTI 类 型 是 "com.yourdomain.app.type" 
-(void)shareToPublicCustomData:(NSData *)data { 
[[UIPasteboard generalPasteboard] 
setData:data 
forPasteboardType:@"com.yourdomain.app.type"]; © 
} 


// 共 享 至 自 定义 命名 的 剪贴 板 
-(void)sharePrivatelyCustomData:(NSData *)data { 
UIPasteboard *appPasteboard - [UIPasteboard 
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pasteboardwWithName:@"myApp" 
create:YES]; @ 


[appPasteboard 
setData:data 
forPasteboardType:Q"com.yourdomain.app.type"]; © 
} 


// 从 公共 的 剪贴 板 读 取 
-(NSArray *)readSharedStrings { 
return [UIPasteboard generalPasteboard].strings; (9 


j 


// 从 命名 的 剪贴 板 读 取 
-(NSData *)readPrivateData { 
UIPasteboard *appPasteboard = [UIPasteboard 
pasteboardWithName:@"myApp" 
create: YES]; 


return [appPasteboard 
dataForPasteboardType:Q"com.yourdomain.app.type"]; € 
j 


Q 设置 已 知 类 型 kUTTypeRTF 的 二 进 制 数据 。 

@ 可 以 为 kuTTypePlainText 类 型 设置 纯 字 符 串 ， 也 可 以 使 用 string 属性 。 对 NSString 对 
象 的 数组 而 言 ， 使 用 strings 属性 也 是 可 行 的 。 

Q 为 自 定义 类 型 设置 二 进 制 数据 。 

O 获取 具有 给 定名 称 的 剪贴 板 。 若 不 存在 ， 新 建 一 个 。 

O 设置 自 定义 剪贴 板 的 数据 。 

@ 检索 存储 在 strings 属性 中 的 字符 串 数 组 。 

@ 在 自 定 义 剪贴 板 〈 非 公开 ) 中 检索 自 定义 类 型 的 数据 。 


与 深层 链接 相 比 ， 剪 贴 板 具 有 以 下 优点 。 

。 它 具 有 支持 复杂 数据 〈 如 图 像 ) 的 能 力 。 

。 它 支 持 在 多 种 形式 中 表示 数据 ， 这 些 形 式 可 以 基于 目标 应 用 的 功能 来 选择 。 例 如 ， 消 息 
应 用 可 以 使 用 纯 文 本 格式 ， 邮 件 应 用 可 以 使 用 来 自 同一 剪贴 板 项 目的 富 文本 格式 。 

。 即使 应 用 关闭 后 ， 剪 贴 板 内 容 仍然 会 保留 。 

然而 ， 与 深层 链接 相 比 ， 使 用 剪贴 板 的 明显 缺点 是 共享 数据 的 格式 不 是 任何 标准 格式 。 因 

此 ， 如 果 不 能 定义 两 个 应 用 之 间 的 数据 契约 ， 那 么 剪贴 板 将 不 能 用 于 通用 共享 。 

如 例 8-2 所 示 ， 使 用 多 个 选项 可 以 共享 纯 文 本 数据 ， 有时， 选择 使 用 哪 一 种 格式 会 令 人 感 

到 困惑 。 

与 深层 链接 不 同 ， 剪 贴 板 不 能 用 于 检测 是 否 安装 了 目标 应 用 。 此 信息 有 助 于 提供 更 好 的 用 

户 体验 ， 例 如 ， 如 果 尚 未 安装 应 用 ， 那 么 可 以 提示 用 户 进行 安装 。 

此 外 ， 与 深层 链接 不 同 ， 任 何 应 用 都 可 以 访问 剪贴 板 ， 因 此 它 会 带 来 深层 的 安全 问题 。 

使 用 剪贴 板 时 ， 你 应 该 遵循 以 下 的 最 佳 实践 。 
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剪贴 板 本 质 上 是 由 剪贴 板 服务 进行 调解 的 进程 间 通 信 。IPC 的 所 有 安全 规则 都 适用 ( 例 
如 ， 不 发 送 任何 安全 数据 、 不 信任 任何 传人 数据 ) 。 

因为 不 能 控制 哪个 应 用 会 访问 剪贴 板 ， 所 以 使 用 时 总 是 不 安全 的 ， 除 非 数 据 被 加 密 。 

不 要 在 剪贴 板 中 使 用 大 量 数 据 。 虽 然 剪贴 板 支持 交换 图 像 以 及 多 种 格式 ， 但 请 记 住 ， 每 
个 条 目 不 仅 消耗 内 存 ， 也 需要 额外 的 时 间 来 读 写 。 

当 应 用 将 使 用 UIApplicationDidEnterBackgroundNotification 通知 或 UIApplicationWi 
llResignActiveNotification 通知 进入 后 台 时 ， 清 除 剪 贴 板 。 更 好 的 做 法 是 ， 你 可 以 实 
现 UIApplicationDelegate 相应 的 回调 方法 。 

通过 将 items 设置 为 nil， 你 可 以 清除 剪贴 板 ， 如 下 所 示 : 


myPasteboard.items = nil; 


为 了 防止 任何 类 型 的 复制 /粘贴 ， 继 承 UITextView, Jf E canPerformAction 的 copy: 动 
作 中 返回 NO, ° 


8.3 HEAR 














我 们 前 面 探 索 的 两 个 选项 一 一 日 定义 URL scheme 和 剪贴 板 一 一 完全 是 机 器 驱动 的 ， 不 受 
人 为 控制 。 终 端 用 户 不 能 选择 要 进入 哪个 应 用 或 数据 将 用 于 哪个 应 用 。 
为 了 填补 这 个 空白 ，iOS 提供 了 儿 个 选项 用 于 与 特定 应 用 共享 文档 。 源 应 用 生成 要 共享 


的 数据 ， 用 户 选择 将 要 使 用 数据 的 应 用 。 图 8-2 显示 了 使 用 文档 共享 选项 的 WhatsApp 和 
Photos 应 用 。 可 用 的 应 用 列表 取决 于 所 选 的 选项 ， 我 们 将 在 下 面 进行 探讨 。 





























Cancel 1 Photo Selected 





AirDrop. Tap to turn on Wi-Fi and 
Bluetooth to share with AirDrop. 







'orwar: 





Copy Slideshow Ig 
act Wallpaper 








图 8-2. WhatsApp 和 Photos 应 用 分 别 分 享 消息 和 照片 





iE 3: Stack Overflow, “How to Disable Copy, Cut, Select, Select All in UITextView" (http://stackoverflow.com/ 





a/1429320). 
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8.3.1 文档 交互 


B iOS 3.2 起 就 已 经 可 用 的 UIDocumentInteractionController 类 允许 应 用 利用 设备 上 的 其 
他 应 用 打开 文档 。 它 还 支持 预览 、 打 印 、 邮 寄 和 复印 文档 。 











UIDocumentInteractionController 不 是 UIViewController 的 子 类 。 你 必须 配 
置 一 个 视图 控制 器 来 预览 文档 。 





控制 器 的 使 用 涉及 两 个 方面 : 发 布 者 和 用 户 。 

作为 发 布 者， 你 的 应 用 要 发 布 即将 被 查看 的 文档 。 控 制 器 负责 加 载 目标 应 用 ， 让 内 容 对 用 
户 应 用 可 用 ， 最 后 还 要 让 用 户 回 到 主 应 用 。 本 节 的 “发 布 者 ”中 介绍 了 发 布 者 方面 的 典型 
代码 。 

作为 用 户 ， 你 的 应 用 的 职责 包括 处 理 文 档 (并 呈现 结果 )， 还 负责 执行 一 些 清理 工作 ， 如 
本 市 的 “消费 者 ”所 示 。 

图 8-3 说 明了 文档 共享 期 间 所 要 经 过 的 步骤 。 




















操作 系统 将 PDF 复制 
到 目标 应 用 的 收 件 箱 





操作 系统 调用 
openURL : 回调 


应 用 加 载 PDF 





应 用 提示 PDF 
即将 打开 


目标 应 用 处 理 
OpenURL : 








iiie 








图 83: 完整 的 文档 共享 过 程 


1. 发 布 者 

作为 发 布 者 ， 应 用 可 以 预览 或 打开 文档 。UIDocumentInteractionController 通过 UIDocument- 
InteractionController Delegate 作为 主 应 用 的 委托 ， 指 定 父 视图 控制 器 显示 预览 窗口 。 视 
图 控制 器 还 需要 文档 的 路 径 ， 以 及 与 文档 类 型 相关 联 的 统一 类 型 标识 符 。 








统一 类 型 标识 符 
统一 类 型 标识 符 (uniform type identifier, UTI) 是 用 来 唯一 标识 某 一 项 目 类 型 的 文本 
"AGE. ABA UTI 用 来 标识 公共 系统 对 象 。 例 如 ， 文 档 的 public.document, JPEG 
图 像 的 public.jpeg 和 纯 文 本 的 public.plain-text, 
有 一 个 规定 允许 第 三 方 开发 人 员 添 加 自己 的 UTI， 用 于 特定 应 用 或 专 有 用 途 。 例 如 ， 
用 于 PDF 文档 的 com.adobe.pdf 和 用 于 Apple Keynote 文档 的 com.apple.keynote.key, 











委托 获取 一 个 回调 以 确定 是 否 可 以 执行 特定 操作 。 这 些 操作 默认 情况 下 包括 copy:、 
print: 和 saveToCameraRoll:。 在 预览 /打开 的 UI 即将 呈现 之 前 以 及 呈现 之 后 ， 它 都 会 获 
得 相应 的 回调 。 


例 8-3 显示 了 对 共享 文档 进行 预览 和 打开 的 示例 代码 。 
例 8-3 文档 交互 一 一 发 布 者 


#import «MobileCoreServices/MobileCoreServices.h» @ 





(interface HPDocumentViewerViewController 
«UIDocumentInteractionControllerDelegate» @ 


(property (nonatomic, strong) UIDocumentInteractionController *docController; 
Qend 
(implementation HPDocumentViewerViewController 


-(UIViewController *)documentInteractionControllerViewControllerForPreview: 
(UIDocumentInteractionController *) controller { © 
return self; 


} 


-(NSURL *)fileInDocsDirectory:(NSString *)filename { 
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, 
NSUserDomainMask, YES); 
NSString *docsDir = [paths firstObject]; 
NSString *fullPath = [docsDir stringByAppendingPathComponent: filename] ; 


return [NSURL fileURLWithPath: fullPath] ; 
} 


-(void)configureDIControllerWithURL:(NSURL *)url 
uti:(NSString *)uti ( @ 


UIDocumentInteractionController controller - [UIDocumentInteractionController 
interactionControllerWithURL:url]; © 
controller.delegate - self; 
controller.UTI = uti; © 
self.docController = controller; @ 
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N 





-(IBAction)previewDocument { 
NSURL *fileURL = [self fileInDocsDirectory:@"sample.pdf"]; © 


if(fileURL) { 
[self configureDIControllerWithURL:fileURL uti:kUTTypePDF]; 
[self.docController presentPreviewAnimated:YES]; © 


j 


-(IBAction)openDocument { 
NSURL *fileURL = [self fileInDocsDirectory:@"sample.pdf"]; 


if (fileURL) { 
[self configureDIControllerWithURL:fileURL uti:kUTTypePDF]; 


[self.docController presentOpenInMenuFromRect:self.view.frame 
inView:self.view animated:YES]; (9 
} 
} 
@end 


@ UlDocumentInteractionController 类 、 相 关 的 类 型 以 及 常量 定义 在 MobileCoreServices 中 。 

Q 控制 器 需要 UIDocumentInteractionControllerDelegate。 让 视图 控制 器 实现 协议 。 

© 虽然 所 有 的 方法 都 是 可 选 的 ， 但 你 必须 实现 方法 documentInteractionControllerViewCo 
ntrollerForPreview:， 因 为 该 方法 提供 了 将 要 展示 子 视图 控制 器 的 UIViewController, 

O 辅助 方法 使 用 内 容 URL fü UTI 类 型 配置 控制 器 。 

© 获取 URL 指向 的 控制 器 的 引用 。 

@ 指定 委托 和 UTI 类型。 

O 设置 对 控制 器 的 强 引 用 ， 确 保 控制 器 不 会 过 早 地 被 释放 。 

© 请 注意 ，UIDocumentInteractionController 对 象 引 用 的 URL 必须 是 操作 系统 可 访问 的 。 
如 果 需 要 ， 那 么 下 载 文件 的 内 容 ， 并 使 用 本 地 (文件) URL 进行 引用 。 

© 使 用 presentPreviewAnimated: 预览 文档 。 如 图 8-4 所 示 ， 在 应 用 中 展示 有 动画 效果 的 
视图 控制 器 。 

@ 使 用 presentOpenInMenuFromRect:inView:animated: 显示 “在 …… 中 打开 ”的 菜单 ， 并 
让 用 户 选 择 某 一 应 用 打开 文档 。 图 8-5 显示 了 应 用 中 的 菜单 。 


UIDocumentInteractionController 需要 一 个 NSURL 来 读 取 内 容 ， 它 必须 指 
向 一 个 使 用 文件 scheme 的 本 地 文件 。 任 何其 他 scheme 将 引发 异常 ， 最 终 有 
可 能 导致 应 用 月 并。 
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CHAPTER 1 
Performance in Mobile Apps 


With this book in your hand, 1 can presume that you are an (OS developer and have 
been writing native OS applications for a substantial about of time, and that you are 
now willing to take leap from y t IOS developer to top of the league. 

In this chapter we get a head-start on what performance is all about and what does it 
take to write a high performance iOS app. 





Defining Performance 


From technical standpoint, performance is, strictly speaking, a very vague term. When. 
someone says - this is a high performing application, we dont really know what he is 
really talking about. Does it mean that the app uses less memory, does it save you money. 
‘on network usage or does it allow you to work fluidly - the meaning and implications 
can be many. 


For me, when the word performance pops up, I relate it to one or more of the consid- 














l^ 


图 8-4. 预览 PDF 文档 〈 显 示 的 视图 控制 器 ) 





AirDrop. Tap to turn on Wi-Fi and 
Bluetooth to share with AirDrop. 


26 


Openin Openin Drive Open in Open in 
Dropbox iBooks Chrome 


Cancel 











& 8-5. PDF 文档 中 的 “在 …… 中 打开 ”菜单 
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2. 消费 者 
作为 文档 的 消费 者 需要 两 个 基本 步 又 : 注册 应 用 支持 的 文件 类 型 ， 然 后 处 理 文 档 内 容 。 你 
可 以 支持 一 个 或 多 个 iOS 系统 定义 的 类 型 ， 也 可 以 注册 新 类 型 ， 注 册 的 新 类 型 有 助 于 在 同 
一 公司 或 其 他 公司 的 一 系列 应 用 之 间 进 行 共享 。 
要 想 注 册 应 用 支持 的 文件 类 型 ， 你 必须 在 应 用 的 Info.plist 中 的 文档 类 型 部 分 配置 以 下 详 
细 信 息 。 
。 名 称 

你 想 提 供 的 人 类 可 读 的 名 称 。 

标准 的 统一 类 型 标识 符 “ 之 一 (如 com.adobe.pdf) 或 自 定义 UTI, 
。 图 标 

如 果 与 应 用 图 标 不 同 ， 则 需要 一 个 与 文档 相关 联 的 图 标 。 
。 属性 

作为 可 选项 ， 你 可 以 配置 其 他 文档 类 型 属性 。 
图 8-6 显示 了 应 用 的 文档 类 型 部 分 。 请 注意 ， 类 型 已 经 被 设置 为 com.adobe.pdf， 这 是 一 种 
预定 义 类 型 。 随 意 选 择 一 个 自 定义 名 称 ， 以 便 在 各 组 应 用 中 共享 自 定 义 类 型 。 必 须 将 相同 
的 值 用 作 UIDocumentInteractionController 对 象 的 UTI 属性 。 



























































Y Document Types (1) 
HighPerf PDF &e 


Name HighPerf PDF 
No 


image Types com.adobe.pdf 
specified 
Icon 
Add icons here 
+ 
b Additional document type properties (0) 


+ 
8-6: 配置 文档 类 型 以 处 理 PDF 文档 


使 用 此 设置 ， 如 果 在 Safari 中 打开 URL: https://bitcoin.org/bitcoin.pdf， 然 后 点 击 “ 在 …… 
中 打开 ”菜单 ， 那 么 它 会 显示 出 我 们 的 应 用 ( 见 图 8-7). 























注 4: 有 关 完 整 列表 ， 请 参阅 iOS 开发 者 库 (https://developer.apple.com/library/ios/documentation/Miscellaneous/Re- 
ference/UTIRef/Articles/System-DeclaredUniformTypeldentifiers.html#//apple_ref/doc/uid/TP40009259-SW 1). 
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AirDrop. Tap to turn on Wi-Fi and 
Bluetooth to share with AirDrop. 


cms 
High 
Performance 


Open in Opén in pen 
HPerf Apps — iCan}Print Mobilel 





Cancel 


图 8-7: 查看 PDF 文档 时 ,在 Safari 中 的 “在 …… 中 打开 ”菜单 

















现在 ， 我 们 必须 处 理应 用 的 委托 回调 openURL: sourceApplication:annotation:。 如 图 8-3 
所 示 ， 共 享 文档 被 复制 到 应 用 的 收 件 箱 文件 夹 中 。 传 递 给 回调 的 url 是 一 个 指向 文件 的 文 
fF URL。 如 果 用 户 对 同一 应 用 多 次 共享 同一 文档 ， 那 么 操作 系统 会 创建 文档 的 多 个 副本 ， 
且 每 次 都 有 新 的 可 用 URL. fil 8-4 显示 了 处 理 文档 的 典型 代码 。 


例 8-4 文档 交互 一 一 消费 者 


@implement HPAppDelegate 














-(BOOL)application:(UIApplication *)application 
OpenURL: (NSURL *)url sourceApplication:(NSString *)src 
annotation: (id)annotation { 


DDLogDebug("%s src: %@, url: %@", ^— PRETTY FUNCTION _, src, url); 
return YES; 


} 
@end 


如 果 查 看 图 8-8 中 应 用 委托 的 日 志 ， 你 会 注意 到 ， 即 使 在 同一 个 应 用 中 打开 同一 文档 ， 每 
次 的 url 也 是 不 同 的 。 
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Clear 


10:27 PM 


7 © 88% G+ 


Debug Log Refresh 


wi 
W] 


] [applicationDidBecomeActive] cz called 
[applicationWillResignActive] called 


[W] [applicationDidEnterBackground] called 
[W] [shouldSaveApplicationState] called 
[W] [willEncodeRestorableStateWithCoder] 


called 


[W] [applicationWillEnterForeground] called 

[W] [openURL] src: (null), url: file:///private/var/ 
mobile/Containers/Data/Application/E61BE37E- 
D26A-4A0C-A222-43F32BF28560/Documents/ 
Inbox/bitcoin.pdf 
[W] [applicationDidBecomeActive] called 
[W] [applicationWillResignActive] called 

[W] [applicationDidEnterBackgroung] called 
[W] [shouldSaveApplicationState] called 
[W] [willEncodeRestorableStateWithCoder] 


called 


[W] [applicationWillEnterForeground] called 

[W] [openURL] src: (null), url: file:///private/var/ 
mobile/Containers/Data/Application/E61BE37E- 
D26A-4A0C-A222-43F32BF28560/Documents/ 
Inbox/bitcoin-1.pdf 
[W] [applicationDidBecomeActive] called 


Chapters 


ep 


Debug Log Settings 














图 8-8: 应 用 委托 回调 的 调试 日 志 





使 用 UIDocumentInteractionController 共享 文档 时 需要 遵循 的 最 佳 实践 ， 与 我 们 之 前 讨论 
我 们 更 应 该 关心 另 一 个 问题 


如 上 所 述 ， 使 用 UIDocumentInteractionController 会 导致 文档 被 复制 


共享 选项 时 的 最 佳 实践 类 似 。 此 外 ， 


夹 中 。 因 此 ， 应 用 有 责任 删除 文件 并 清理 文件 夹 。 生 成 的 文人 





不 要 忘记 删 除 文 件 o 


8.32 活动 





UTI 在 过 去 可 以 很 好 地 工作 。 然 而 ， 云 服务 和 六 
得 紧张 。 


因此 ，UTI 和 远程 URL 之 间 的 关系 变 


UIActivityViewController 提供 了 一 


关 的 操作 。 它 在 iOS 6 中 被 引入 。 
































F 交 媒体 的 兴起 使 还 和 





到 应 用 的 收 件 箱 文件 


F 由 用 户 应 用 所 有 。 完 成 后 ， 


实体 优 于 本 地 文件 。 


个 统一 的 服务 接口 ， 以 便 共享 和 执行 与 应 用 中 数据 有 


使 用 UIActivityViewController 比 使 用 UIDocumentInteractionController 更 容易 、 更 灵活 。 





UIDocumentInteractionController 只 人 允许 文件 URL， 但 使 月 


以 共享 以 下 一 种 或 多 种 类 型 。 
e NSString 
任何 字符 串 都 可 以 共享 。 











H UIActivityViewController 可 
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e NSAttributedString 


可 以 共享 格式 化 文本 或 富 


。 NSURL 
任何 URL 都 可 以 共享 。 
选择 按 原样 共享 URL, 


e UIImage 


如 果 提 供 了 图 像 ， 那 么 


e ALAsset 
这 展示 由 照片 应 用 管理 


e UIActivityItemSource 





文本 。 


根据 目标 应 用 的 不 同 而 使 用 不 同 的 URL。 邮 件 或 消息 应 用 可 以 
而 阅读 器 或 云 服 务 应 用 可 能 会 尝试 获取 内 容 并 进行 处 理 。 





图 片 也 可 以 保存 到 相机 胶卷 ， 分 配给 联系 人 或 打印 。 





的 照片 或 视频 ， 可 以 与 目标 应 用 共享 。 





符合 此 协议 的 对 象 可 以 共享 。 这 有 助 于 创建 可 以 跨 应 用 共享 的 自 定义 对 象 。 





要 想 将 UIImage 保存 到 相机 胶卷 或 使 用 ALAsset， 应 用 需要 获得 访问 照片 的 
权限 。 如 果 应 用 从 未 询问 过 ， 那 么 系统 会 提示 用 户 授予 权限 。 如 果 用 户 之 前 
已 经 拒绝 或 授予 访问 权限 ， 则 不 会 出 现 权限 提 示 。 





有 两 种 类 型 的 活动 : 操作 和 共享 ( 见 图 8-9)。 共 享 从 第 三 方 应 用 (如 Facebook, Twitter, 
Vimeo 等 ) 启动 UI， 而 操作 主要 涉及 内 置 的 应 用 (照片 、 打 印 机 、 剪 贴 板 /复制 、Safari、 


联系 人 等 )。 此 外 ， 还 有 


AirDrop 支持 图 像 。 苹 果 公 司 的 开发 者 网 站 (https://developer. 





apple.com/library/ios/documentation/UIKit/Reference/UIActivity_Class/index.html#//apple_ref/ 
doc/constant_group/Built_in_Activity_Types) 上 提供 了 内 置 操作 类 型 的 完整 列表 。 











(9) AirDrop. Tap to turn on Wi-Fi and 


Bluetooth to share with AirDrop. 


ooog}. 


Message Mail Twitter Facebook 


Save Image Assign to Add to Copy 
Contact Reading List 








& 8-9; 使 用 UIActivityViewController 一 一 操作 和 共享 活动 
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例 8-5 显示 了 在 应 用 中 启用 活动 视图 控制 器 的 典型 代码 。 你 可 以 选择 排除 一 个 或 多 个 活动 。 
此 外 ， 通 过 实现 自 定义 UIActivity， 你 可 以 让 用 户 将 内 容 共 享 到 同一 个 应 用 ，UIActivity 
是 一 个 必须 被 子 类 化 的 抽象 类 。 


例 8-5 使 用 UIActivityvViewController 共享 数据 


-(void)shareSomeContent { 
NSString *text = @"Text to share"; 
NSURL *url = [NSURL URLWithString:@"http://github.com"]; 
UIImage *image = [UIImage imageNamed:"blah"]; @ 








NSArray *items = @[text, url, image]; 

UIActivityViewController *ctrl = [[UIActivityViewController alloc] 
initWithActivityItems:items 
applicationActivities:nil]; € 


ctrl.excludedActivityTypes = @[ UIActivityTypePostToFacebook ]; © 
[self presentViewController:ctrl animated:YES completed:nil]; €) 


O 要 共享 的 几 个 项 目 一 一 字符 串 、URL 和 图 片 。 

@ 使 用 活动 项 目 实 例 化 UIActivityViewController。 此 示例 没有 配置 任何 applicationActivities, 
如 果 需 要 ， 它 们 必须 是 UIActivity 的 子 类 对 象 。 

© 排除 不 允许 的 活动 类 型 一 一 此 处 已 排除 在 Facebook. 上 发 帖子 。 

O 最 后 ， 呈 现 视图 控制 器 ， 使 用 模式 (如 这 里 演示 的 ) 或 使 用 navigationController。 


就 与 其 他 应 用 共享 内 容 而 言 ， 活 动 非常 灵活 、 可 扩展 ， 而 且 功 能 强大 。 但 性 能 和 安全 问题 
是 之 前 讨论 的 其 他 数据 共享 方式 中 存在 的 问题 的 集合 。 


























共享 钥匙 串 
共享 钥匙 事 是 在 应 用 间 安 全 共享 数据 的 另 一 种 选择 。 只 有 属于 相同 群 组 ID， 且 使 用 相 
同 证 书签 名 的 应 用 才能 共享 数据 。 
在 你 的 所 有 应 用 中 实现 单 点 登录 的 唯一 方法 就 是 使 用 共享 钥匙 囊 。 
要 实现 在 同一 发 布 者 (相同 的 签名 证 书 ) 的 应 用 之 间 共 享 数据 ， 这 是 唯一 的 方式 ， 且 
不 需要 从 用 户 正在 使 用 的 应 用 调用 其 他 应 用 。 
因为 数据 是 加 密 的 ， 所 以 它 应 该 是 存储 安全 信息 的 地 方 ， 如 凭证 、 信 用 卡号 (BRE 
有 CVV) 等 。 避免 在 大 量 通用 、 非 安全 数据 中 刷新 ， 因 为 这 样 的 访问 会 比 对 末 加 密 数 
据 的 访问 慢 。 











8.4 iOS 8 扩展 


iOS 8 引入 了 四 个 新 选项 ， 用 于 在 应 用 之 间 共 享 内 容 。 这 四 个 选项 被 用 在 应 用 扩展 (在 第 6 
章 中 简要 介绍 过 ) 的 更 广泛 类 别 之 中 。 


如 果 打 开 项 目 并 点 击 加 号 图 标 (+) 来 添加 目标 (ILE 8-10), ， 你 应 该 可 以 在 iOS 下 找到 一 
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个 新 的 应 用 扩展 条 目 ， 该 条 目 会 提供 如 





图 8-11 所 示 的 选项 。 



























































al < 加 HighPerformance 
I General Capabilities Info Build Settings Build Phases Build Rules 
PROJECT t Q 
n HighPerformance 
b» Target Dependencies (0 items) 
TARGETS 
> Check Pods Manifest.lock 
[HighPerformanceTests 
> Compile Sources (39 items) 
> Link Binary With Libraries (17 items) 
> Copy Bundle Resources (14 items) 
> Fabric 
> Copy Pods Resources 
©- © 
&] 8-10: Xcode 一 Project — Add Target 
Choose a template for your new target: 
iOS Y / : » . 
pa 3898) nb 
Application j = CU») f 
Framework & Library ! 
Noa Aaa Action Custom Document Photo Editing 
bp Extension Keyboard Provider Extension 
Other 
Apple Watch t 17^ 
OS X - = 
Application Share Extension Today 
Framework & Library Extension 
Application Extension 
System Plug-in Share Extension 
Other This template builds a Share application extension. 
Cancel — Next 














&] 8-11; 选择 iOS — Application Extension 





。 操作 扩展 
分 享 扩展 
文档 提供 者 扩展 
应 用 群 组 


从 实现 的 角度 看 ， 我 们 对 它们 并 不 陌生 。 在 上 一 节 中 ， 我 们 已 经 讨论 过 通过 动作 进 


在 所 有 的 选项 中 ， 只 有 以 下 儿 项 是 用 来 实现 数据 共 


m 
Eq 
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JJ. 
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使 用 UIActivityViewController 实现 共享 ， 以 及 使 用 UIDocumentInteractionController SZ 

现 文档 共享 。 

与 之 前 相 比 ， 本 市 介绍 的 较为 新 鲜 的 内 容 是 整体 的 探索 、 实 施 的 难 易 程度 ， 以 及 为 终端 用 

户 提 供 的 选项 。 

在 探索 这 些 共享 数据 的 扩展 之 前 ， 我 们 先 探 讨 在 iOS 8 中 添加 的 一 些 新 类 。 

e NSExtensionContext (http://apple.co/1HejEzB ) 
表示 调用 扩展 时 的 主 应 用 的 上 下 文 。 它 提供 了 输入 项 的 一 个 数组 (例如 ， 共 享 给 应 用 的 
数据 )。 

e NSExtensionItem (http://apple.co/1BFpuls) 
表示 输入 项 数组 中 的 某 一 项 。NSExtensionIten 对 象 是 一 个 不 可 变 集 合 ， 集 合 中 的 
表 了 项 目的 不 同方 面 ， 可 通过 attachments 属性 获得 。 

e NSItemProvider (http://apple.co/1 ECgC4H) 
表示 可 在 NSExtensionItem 对 象 的 attachments 属性 中 找到 的 数据 对 象 ， 如 文本 ， 图 像 
和 URL. fE hasItemConformingToTypeIdentifier: 方法 检查 其 表示 的 UTI 类 型 。 要 检 
索 相 应 类 型 的 数据 ， 请 使 用 方法 LoadItemForTypeIdentifier:options:completionHandl 
er:。 回 调 方法 可 以 在 任何 线程 中 被 调用 一 一 更 改 UI 时 ,不 要 忘记 将 上 下 文 切 换 到 主线 程 。 


8.4.1 配置 操作 扩展 和 共享 扩展 
对 于 操作 和 共享 扩展 而 言 ， 除 了 特定 项 目 ， 二 者 对 所 有 模板 都 是 通用 的 。 
。 元 数据 (Info.plist) 
最 重要 的 两 个 条 目 是 包 显示 名 称 和 NSExtensionItem， 前 者 指 的 是 在 条 目 旁 边 显示 的 名 


称 , 后 者 提供 了 何 时 在 列表 中 显示 此 操作 的 元 数据 。Xcode 将 NSExtension 一 NSExtensio 
nAttributes 一 NSExtensionActivationRule 的 值 设 置 为 TRUEPREDICATE (类 型 为 String), 


实质 上 表示 该 操作 始终 可 用 。 可 以 将 其 更 改 为 Dictionary 类 型 ， 并 使 用 表 8-1 中 列 出 的 
键 来 提供 更 细 粒 度 的 控制 。 


表 8-1: 应 用 扩展 键 
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键 描述 
NSExtensionActivationSupportsText 应 用 支持 文本 (NsString gk NSAttributedString 
类 型 的 值 ) 
NSExtensionActivationSupportsFileWithMaxCount 应 用 支持 任何 文件 的 处 理 (使 用 文件 scheme 
的 NSURL) 
NSExtensionActivationSupportsWebURLWi thMaxCount 应 用 支持 网 页 URL (使 用 http 或 https scheme 
的 NSURL) 
NSExtensionActivationSupportsWebPageWithMaxCount 应 用 支持 网 页 
NSExtensionActivationSupportsImageWithMaxCount 应 用 支持 图 像 (UIImage fü) 
NSExtensionActivationSupportsMovieWithMaxCount 应 用 支持 视频 
NSExtensionActivationSupportsAttachmentsWithMinCount ”扩展 要 激活 的 最 小 附件 数 (默认 为 1) 
NSExtensionActivationSupportsAttachmentsWithMaxCount ”扩展 要 激活 的 最 大 附件 数 (默认 为 long max fü) 
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正 值 表示 可 以 为 特定 类 型 共享 的 条 目的 最 大 值 。 例 如 ，NSExtensionActivationSuppo 
rtsImageWithMaxCount 的 值 为 2， 这 表示 最 多 可 以 共享 两 个 图 像 。 缺 少 键 或 值 为 0 则 
意味 着 扩展 不 支持 此 特定 类 型 。 要 声明 一 个 更 复杂 的 定义 ， 你 可 以 使 用 NSPredicate- 
compilable 结构 。 请 参阅 “应 用 扩展 编程 指南 ” (http://apple.co/ILDgW9X) 中 的 “为 共 
享 或 操作 扩展 声明 其 支持 的 数据 类 型 ”部 分 。 

。 目标 产品 名 称 
当 创 建新 扩展 时 ， 使 用 产品 名 称 字段 中 提供 的 名 称 创 建新 目标 。 

8.4.2 ”操作 扩展 

在 使 用 UIActivityViewController 时 ， 操 作 扩 展 允 许 你 将 视图 控制 器 添加 到 操作 部 分 。 


iOS 7 捆绑 了 来 自 其 他 应 用 的 预定 义 操 作 列表 ， 但 是 没有 办 法 增加 更 多 的 ，iOS 8 更 改 了 这 
一 点 ， 















































MZ 


当 你 创建 操作 扩展 时 ，Xcode 将 创建 以 下 的 附加 内 容 。 
。 故事 板 主 界面 
当 用 户 选 择 该 操作 时 ， 故 事 板 UI 即将 显示 出 来 。 
e ActionViewController 类 
支持 故事 板 的 视图 控制 器 类 。 
作为 一 个 视图 控制 器 ， 它 有 和 其 他 控制 器 一 样 的 生命 周期 (回忆 一 下 之 前 的 图 6.1)。 
例 8-6 展示 了 泻 染 源 应 用 共享 图 像 的 典型 代码 。 



























































例 8-6 ”操作 一 一 从 共享 数据 演 染 图 片 
- (void)viewDidLoad { 
[super viewDidLoad]; 
BOOL imageFound - NO; 
for(NSExtensionItem *item in self.extensionContext.inputItems) ( @ 
for(NSItemProvider *itemProvider in item.attachments) ( @ 
if([itemProvider 
hasItemConformingToTypeIdentifier:(NSString *)kUTTypeImage]) ( 9 

[self processItem:itemProvider]; 
imageFound = YES; 


break; 
} 
} 
if(imageFound) { 
break; 
} 
} 


} 


-(void)processItem:(NSItemProvider *)itemProvider { 
UIImageView _ weak *imageView = self.imageView; 


[itemProvider loadItemForTypeIdentifier:(NSString *)kUTTypeImage 
options:nil 


completionHandler:^(UIImage *image, NSError *error) ( €) 
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if(image) ( 
[[NSOperationQueue mainQueue] addOperationWithBlock:^[ 
[imageView setImage:image]; 
i; © 


H; 
} 
Q 扫描 所 有 的 扩展 项 。 
e 对 于 每 个 项 目 而 言 ， 扫 描 所 有 的 附件 。 
Q 检查 附件 是 否 为 图 像 类 型 。 
O 如 果 是 ， 请 检索 内 容 。 
O 因为 检索 回调 可 以 在 非 主 线程 上 被 调用 ， 所 以 切换 上 下 文 ， 以 便 使 用 Umage 内 容 更 新 


UIImageView, 


8.4.8 ”共享 扩展 
共享 扩展 与 共享 活动 略 有 不 同 ， 前 者 是 系统 提供 的 UI， 无 法 由 接收 应 用 自 定义 。 
用 户 可 以 在 系统 提供 的 UI 中 访问 共享 扩展 。 在 iOS 中 ， 用 户 点 击 共享 按钮 ， 然 
后 从 显示 的 活动 视图 控制 器 的 共享 区 域 中 选择 共享 扩展 。5 
创建 共享 扩展 时 ，Xcode 将 创建 以 下 的 附加 内 容 。 
。 故事 板 主 界面 
至 少 在 IOS 8.2 之 前 ， 这 没有 什么 意义 。 在 未 来 ， 毕 果 公 司 可 能 会 允许 应 用 提供 自 定义 UL, 
e ShareViewController 类 
这 是 在 iOS 8 中 引入 的 SLComposeServiceViewController 的 子 类 。 虽 然 它 是 一 个 视图 控 
制 器 ， 但 控制 器 配置 的 UI 完全 被 忽略 了 。 
这 个 类 提供 了 以 下 生命 周期 事件 的 挂钩 。 
。 内 容 验 证 
调用 的 第 一 个 方法 是 isCcontentValid。 使 用 NSExtensionContext 验 证 传 入 的 值 ( 见 例 8-6)， 
如 果 数 据 有 效 则 返回 YES， 如 果 无 效 就 返回 NO0。 无 论 值 是 什么 ， 该 活动 将 始终 被 显示 ， 
但 如 果 内 容 无 效 ，Post 按钮 将 被 禁用 ( 见 图 8-12). 
e viewDidLoad 
该 方法 在 验证 初始 内 容 及 加 载 视图 后 被 调用 。 如 果 需 要 的 话 ， 使 用 textview 属性 访问 
UITextView 编辑 器 ， 从 而 对 文本 进行 更 改 。 
。 获取 配置 项 
视图 加 载 后 ，configurationItems 方法 会 被 调用 来 检索 配置 项 。 它 返回 一 个 值 为 0 的 数 
组 或 多 个 SLComposeSheetConfigurationItem ( 子 类 ) 条 目的 数组 。 





































































































iE 5: iOS Developer Library, “Share” (https://developer.apple.com/library/ios/documentation/General/Conceptual/ 
ExtensibilityPG/Share.html). 
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SLComposeSheetConfigurationItem 对 象 使 用 标准 的 UITabtLeview， 以 提供 要 显示 在 共享 
编辑 器 下 面 的 标题 和 值 ( 见 图 8-12). 





























Cancel Post) Cancel 
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Configuration Configuration 














图 8-12: BRAS (£) 和 无 效 内 容 (6) 的 共享 活动 


SLComposeSheetConfigurationltem 对 象 还 提供 了 一 个 tapHandter， 用 户 点 击 配置 项 时 会 
调用 后 者 。tapHandler 可 以 压 入 一 个 视图 控制 器 ， 用 来 显示 对 值 进行 修改 的 一 些 选项 。 
例如 ，Facebook 允许 更 改 图 像 的 相册 、 位 置 和 隐私 值 ( 见 图 8-13 ) 。 
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B 8-13: Facebook 共享 一 一 活动 (£) 和 相册 配置 (6) 
。 视图 控制 器 生命 周期 方法 
随后 调用 viewWillAppear 和 viewDidAppear 方法 。 


。 内 容 验 证 
在 显示 视图 后 立即 执行 内 容 验证 的 最 后 一 段 。 
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内 容 变化 的 验证 

每 当 用 户 更 改编 辑 器 中 的 内 容 时 , 调用 isContentvaLid。 扩 展 可 以 调用 validateContent 
方法 来 触发 重新 验证 ， 或 调用 reloadConfigurationItems 方法 来 重新 加 载 配置 项 。 它 还 
可 以 实现 charactersRemaining 方法 ， 以 返回 一 个 非 负 值 ， 表 示 剩 余 的 字符 数 〈 图 8-14 

展示 了 一 个 显示 该 值 的 Twitter 共享 ) 。 














Cancel Twitter 
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图 8-14; Twitter 共享 ， 展 示 剩余 的 字符 数 


取消 通知 
如 果 用 户 点 击 取消 按钮 ，didselectCancel 会 被 调用 。 
发 布 通知 
如 果 用 户 点 击发 布 按钮 ，didSelectPost 会 被 调用 。 
使 用 共享 扩展 时 ， 请 注意 以 下 几 点 。 
。 所 有 方法 都 是 在 用 户 选 择 活动 后 被 调用 的 。 
。 对 于 取消 和 发 布 通知 ， 调 用 NSExtensionItem 的 completeRequestReturn 
ingItems:completionHandler: 方法 ， 以 表示 活动 的 交互 已 经 完成 了 。 盏 
则 ， 源 应 用 会 处 于 不 可 用 状态 。 作 为 一 种 最 佳 实践 ， 应 该 根据 iOS 8 或 更 
高 版 本 构建 应 用 。 当 想 在 应 用 中 使 用 活动 时 ， 检 测 应 用 是 否 在 10OS 7 或 更 
时 版 本 上 执行 。 如 果 是 ， 请 使 用 自 定义 活动 视图 控制 器 。 如 果 不 是 ， 让 操 
作 系 统 为 你 选择 活动 。 





8.4.4 文档 提供 者 扩展 


文档 提供 者 是 文档 交互 APIBJIOS 8 扩展 版 本 。 要 想 读 取 共享 文档 的 内 容 ， 请 使 
用 UIDocumentPickerViewController。 要 呈现 一 个 UI 来 共享 文档 ， 则 应 该 子 类 化 
UIDocumentPickerViewController, 


使 用 文档 提供 者 需要 iCloud 授权 。 进 入 Project > App 一 iCloud， 选 择 iCloud Documents, 
如 图 8-15 所 示 。 
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PROJECT 
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TARGETS 
HighPerformance Services: Key-value storage 
P^ © iCloud D t 
[^ ]HighPerformanceTests e TION - ocuments 
: CloudKit 
E) actionRead 
E) shareRead Containers: o Use default container 
E) documentRead Specify custom containers 


E) documentReadFilePro... v| iCloud.com.m10v.hperf iCloud.$(CFBundleldentifier) 


Cloudkit Dashboard 





Steps: V Add the "iCloud" entitlement to your App ID 
vV Add the "iCloud containers" entitlement to your App ID 
vV Add the "iCloud" entitlement to your entitlements file 
vV Link CloudKit.framework 














8-15. 应 用 清单 一 一 iCloud 授权 


1. 打开 /导入 文档 

UIDocumentPickerViewController (通常 被 称 为 文档 选择 器 ) 提供 了 一 个 钩子 ， 以 便 与 安装 在 
设备 上 的 其 他 文档 提供 者 进行 交互 。 文 档 选 择 器 可 以 在 打开 /导入 模式 或 导出 模式 下 工作 。 
该 方法 类 似 于 文档 交互 提供 者 的 方法 ， 只 是 前 者 是 从 “在 …… 中 打开 ”的 菜单 进入 的 。 
此 ， 用 户 无 需 立 即 进入 提供 文档 的 应 用 (如 浏览 器 、Google Drive, Dropbox 等 )， 而 是 直 
接 在 相关 应 用 中 导入 文档 并 继续 操作 。 

图 8-16 显示 了 使 用 文档 交互 方式 编辑 文档 和 使 用 文档 选择 器 编辑 文档 时 ， 每 个 步骤 的 差别 。 
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图 8-16: 使 用 活动 或 文档 交互 〈 左 ) 和 使 用 文档 选择 器 〈 右 ) 编辑 文档 
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UIDocumentPickerViewController 对 象 需要 配置 以 下 项 目 。 
。 文档 类 型 
编辑 器 应 用 可 以 支持 的 UTI 类 型 。 
。 模式 
必须 被 配置 为 Open 或 Import, 
。 委托 
用 户 选 择 文档 时 ，UIDocumentPickerDelegate 类 型 的 委托 会 发 生 响 应 。 在 用 户 取消 选 
择 时 ， 它 也 可 能 (可 选 ) 响应 。 


例 8-7 显示 了 从 其 他 文档 提供 者 打开 /导入 文档 的 典型 代码 。 
Gil 8-7 文档 提供 者 一 一 打开 /导入 


@interface HPDocumentEditorViewController 
: UIViewController «UIDocumentPickerDelegate» @ 
Qend 





HE 


























(implementation HPDocumentEditorViewController 
-(IBAction)openButtonWasClicked:(id)sender { 
NSArray *types - Q[ 
(NSString *)kUTTypeImage 


pne 


UIDocumentPickerViewController *dpvc - 
[[UIDocumentPickerViewController alloc] 
initWithDocumentTypes:types 
inMode:UIDocumentPickerModeImport]; © 
dpvc.delegate = self; @ 
[self.navigationController presentViewController:dpvc 
animated:YES completion:nil]; G 


j 


-(void)documentPicker: (UIDocumentPickerViewController *)controller 
didPickDocumentAtURL:(NSURL *)url { @ 
NSData *data = [NSData dataWithContentsOfURL:url]; @ 
// 处 理 数据 ,在 编辑 器 中 泻 染 ,让 用 户 编辑 








- (void)documentPickerWasCancelled:(UIDocumentPickerViewController *)controller { @ 


// 或 许可 以 展示 一 条 信息 ,表明 用 户 没 有 选择 文件 



































} 
Qend 


Q 编辑 器 应 用 的 视图 控制 器 遵守 UIDocumentPickerDelegate 协议 。 

Q 编辑 器 可 以 处 理 的 UTI 类 型 。 

@ UlDocumentPickerViewControLler 对 象 使 用 UTI 类 型 进行 配置 日 处 于 UIDocumentPickerModeImport 
模式 。 

O 指定 委托 (这 是 必需 的 步骤 ) 。 














O 呈现 视图 控制 器 。 

Q 当 用 户 选择 文档 时 ， 调 用 委托 回调 方法 。 

O url 是 本 地 文件 URL。 文 档 的 内 容 将 复制 到 应 用 的 tmp/DocumentPickerIncoming 文件 夹 中 。 
O 当 用 户 取消 选择 文档 时 ， 调 用 委托 回调 方法 。 


使 用 文档 选择 器 时 的 用 户 导航 如 图 8-17 所 示 。 
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图 8-17: 文档 导入 的 用 户 导 航 : (1) 默认 情况 下 打开 iCloud， 选 择 位 置 以 便 选择 提供 者 ，(2) 选择 提 
供 者 ，(3) 选择 文档 (4) 在 编辑 器 应 用 中 处 理 文档 


2. 提供 文档 
要 想 成 为 提供 文档 的 数据 源 ， 你 需要 以 下 步骤 。 


(1) 创建 UI 以 帮助 用 户 选 择 文档 。 
(2) 将 文档 的 内 容 传递 给 编辑 器 应 用 。 
要 想 创 建 UI， 视 图 控制 器 必须 是 UIDocumentPickerExtensionViewController 的 子 类 。 当 用 

















户 选 择 此 应 用 作为 位 置 时 ， 该 视图 控制 器 提供 的 UI 在 编辑 器 应 用 中 是 可 用 的 ( 见 图 8-17). 
当 用 户 选 择 要 使 用 的 文档 时 ， 如 果 需 要 ， 视 图 控制 器 必须 从 服务 器 下 载 内 容 ， 并 将 该 文档 
以 文件 URL 的 形式 提供 给 编辑 器 应 用 。 例 8-8 展示 了 选择 器 扩展 的 典型 代码 。 


例 8-8 ”选择 器 扩展 一 一 视图 控制 吕 
@interface HPDocumentPickerViewController 
: UIDocumentPickerExtensionViewController @ 















































Qend 


@interface HPEntry € 

@property (nonatomic, copy) NSString *filename; 
@property (nonatomic, copy) NSString *serverPath; 
@property (nonatomic, assign) NSUInteger *size; 
@property (nonatomic, copy) NSString *uti; 
(property (nonatomic, copy) NSURL *iconURL; 

Qend 


(implementation HPDocumentPickerViewController 


-(void)viewDidAppear:(BOOL)animated { 
[super viewDidAppear:animated]; 
// 从 服务 器 检索 文件 的 元 数据 并 更 新 
//UITableView 是 很 好 的 选择 © 








J 


- (void)tableView:(UITableView *)tableView 
didSelectRowAtIndexPath:(NSIndexPath *)indexPath ( @ 


HPEntry *selected - [self.allFiles objectAtIndex:indexPath.row]; 
// 如 果 需 要 ,从 服务 器 下 载 内 容 O 


NSURL* localFileURL = [self.documentStorageURL 
URLByAppendingPathComponent:selected.filename]; @ 
[self dismissGrantingAccessToURL:localFileURL]; @ 








} 
@end 
Q ET Hl 2k, UIDocumentPickerExtensionViewController 的 子 类 ;向 用 户 提 供 可 用 文 
档 和 目标 的 列表 。 





O 表示 远程 文件 条 目的 模型 类 。 
O 从 服务 器 检索 文件 列表 、 更 新 UI、 等 待 用 户 响应 、 显 示 加 载 指示 器 或 进度 条 一 一 所 有 
与 UI 有 关 的 花哨 东西 。 
O 假设 使 用 了 UITabtLeview， 当 用 户 选 择 文件 时 ， 触 发 委托 回调 。 
O 如 果 文 件 内 容 在 服务 器 上 ， 则 必须 由 应 用 扩展 下 载 。 最 终 的 内 容 必 须 来 自 本 地 文件 URL. 
Q 将 内 容 保 存 到 selfdocumentStorageURL 文件 夹 中 ， 该 URL 是 在 使 用 文件 提供 者 时 设置 的 。 
@ 一 切 都 准备 好 了 。 通 知 操作 系统 ， 编 辑 器 应 用 必须 被 授予 使 用 该 文件 的 权限 。 操 作 系统 
将 文件 复制 到 编辑 器 应 用 对 应 的 文件 夹 中 。 编 辑 器 应 用 中 的 文档 选择 器 (传递 者 ) 会 被 
通知 到 。 我 们 已 经 在 例 8-7 中 讨论 了 整个 故事 的 另 一 半 。 
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8.4.5 ”应 用 群 组 


应 用 扩展 是 一 项 非常 有 趣 的 功能 。 虽 然 扩展 总 是 与 应 用 捆绑 在 一 起 ， 但 它 在 自己 的 进程 中 


运行 ， 并 有 自己 的 数据 沙 箱 。 





所 示 。 








在 一 个 典型 的 场景 中 ， 主 应 用 将 与 应 用 扩展 连接 ， 后 者 还 可 以 与 容器 应 用 连接 ， 如 图 8-18 




















图 8-18: 主 应 用 、 应 用 扩展 和 容器 应 用 之 间 的 通信 (原始 图 像 由 苹果 公司 提供 ) 








到 目前 为 止 ， 我们 讨论 的 所 有 选项 都 是 用 于 在 多 个 应 用 间 共 享 数 据 。 然 而 ， 因 为 应 用 扩展 
在 自己 的 沙 箱 中 运行 ， 所 以 它 因 此 无 法 直接 访问 由 容器 应 用 直接 存储 的 数据 (用 户 默 认 








E. XXR, REXER, Core Data、SQLite， 等 等 )。 沙 箱 结 构 与 图 8-19 所 示 内 容 


类 似 。 

















图 8-19: 应 用 与 应 用 扩展 容器 (原始 图 像 由 苹果 公司 提供 ) 





iOS 7 引入 的 应 用 群 组 功能 允许 创建 一 个 共享 沙 箱 ， 容 器 应 用 和 应 用 扩展 都 可 以 访问 它 。 





此 外 ， 应 用 群 组 支持 在 多 个 应 用 之 间 共 享 数据 





同 的 证 书 进 行 签名 。 





但 与 共享 钥匙 串 类 似 ， 应 用 必须 使 用 相 


配置 应 用 群 组 不 仅 是 为 了 容器 应 用 ， 还 为 了 将 所 有 需要 共享 容器 的 应 用 扩展 
绑 定 起 来 。 可 以 配置 多 个 应 用 群 组 ， 从 而 达到 较为 合理 的 数据 隔离 。 
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要 想 设 置 应 用 群 组 ， 转 到 项 目 清单 ， 选 择 目 标 ， 然 后 在 “功能 ”下 将 “应 用 群 组 ”选项 设 
置 为 “ 开 ”( 见 图 8-20)。 最 后 ， 添 加 你 将 要 使 用 的 一 个 或 多 个 群 组 ID, 
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O jJ 
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ES jJ 
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L]HighPerformanceTests 
E) actionRead Y (86) App Groups 


E) shareRead 
E) documentProvider 
E 


App Groups: 加 i T A 
.) documentProviderFile... FE ps: @ group.com.m1Ov.hperf 2 


Tu © 


Steps: V Add the "App Groups" entitlement to your entitlements file 
vV Add the "App Groups" entitlement to your App ID 
vV Add the "App Groups containers" entitlement to your App ID 











图 8-20. 启用 应 用 群 组 功能 的 Xcode 设置 





现在 你 可 以 使 用 NSuserpefautts 或 NSFiLeManager 来 处 理 共 享 数据 ， 如 示例 8-9 所 示 。 


例 8-9 使 用 应 用 群 组 共享 数据 
-(void)sharedDataUsingAppGroups { 
NSString *sharedGroupId = @"group.com.m10v.hperf"; @ 


NSUserDefaults *defs = [[NSUserDefaults alloc] 
initWithSuiteName:sharedGroupId]; @ 


NSFileManager *fileMgr = [NSFileManager defaultManager ]; 
NSURL *groupFolder = [fileMgr 
containerURLForSecurityApplicationGroupIdentifier:sharedGroupId]; © 
} 


Q 群 组 ID， 必须 与 清单 中 提供 的 匹配 ( 见 图 8-20), 
@ NsUserDefaults, 使 用 initWithSuiteName: 初始 化 器 。 


@ NSFileManager; 使 用 containerURLForSecurityApplicationGroupIdentifier: 方法 获取 
共享 文件 夹 。 





如 果 需 要 访问 网 络 ， 那 么 就 使 用 NSURLSession， 以 便 容器 应 用 和 应 用 扩展 可 
以 访问 传输 的 数据 并 共享 网 络 的 参数 (特别 是 cookie jar). 











8.5 ”小 结 


本 章 探 讨 了 跨 应 用 共享 数据 的 几 个 选项 。 在 所 有 可 用 选项 中 ， 自 定义 URL scheme 是 唯一 
允许 从 Web (通过 Safari W W ak Bate ASK UIWebView) 向 原生 应 用 共享 数据 的 方案 。 但 其 
他 方案 能 够 提供 更 加 丰富 的 UI。 活 动 可 以 弹出 共享 表 ， 让 用 户 选 择 自己 喜欢 的 应 用 来 处 理 
数据 ， 这 在 社交 共享 中 十 分 有 用 。 

对 于 那些 属于 同一 公司 且 使 用 同一 证 书 进行 签名 的 应 用 ， 你 可 以 使 用 共享 钥匙 串 共 享 数 
据 ， 它 可 以 发 挥 强大 的 作用 。 


本 章 末 尾 研 究 了 iOS 8 中 引入 的 应 用 扩展 ， 并 介绍 了 它们 如 何在 应 用 间 帮 助 应 用 扩展 文档 
交互 和 活动 。 应 用 组 可 以 轻松 地 访问 应 用 和 扩展 中 的 共享 数据 。 






































应 用 可 能 会 在 未 知 的 执行 环境 中 运行 ， 并 通过 未 知 的 传输 网 络 交换 数据 ， 因 此 ， 应 始终 将 
安全 性 作为 首要 任务 之 一 ， 以 便 保护 用 户 及 应 用 的 敏感 数据 。 

越狱 和 常规 设备 都 存在 风险 。 例 如 ， 来 自 JosiahsTech 的 YouTube 视频 (https://youtu. 
be/INTpi4NjkCE) 就 演示 了 修改 广 受 欢迎 的 Temple Run 游戏 是 何等 地 简单 。 








RE: 足够 是 不 够 的 
安全 是 一 个 广阔 的 领域 , 仅 任 一 本 书 中 的 几 页 内 容 肯 定 不 足以 深入 详解 该 话题 。 若 想 
更 深层 次 地 研究 ， 可 以 参阅 更 专业 的 内 容 : 
* Jonathan Zdziarski 的 Hacking and Securing iOS Applications: Stealing Data, Hijacking 
Software, and How to Prevent It (O'Reilly, http://amzn.to/1bnf44K ) 
e Charlie Miller 的 《黑客 攻防 技术 宝典 : iOS FRA) ' 
* David Thiel 的 iOS Application Security: The Definitive Guide for Hackers and 
Developers (No Starch Press, http://amzn.to/1Hq15sc) 











不 论 是 通过 代码 的 执行 (例如 ， 从 1024 位 DSA 密 钥 的 加 密 密 钥 转 为 2048 位 RSA 的 加 密 
密 钥 ) 还 是 通过 用 户 干预 〈 例 如 ， 引 入 双 因 素 认 证 或 应 用 PIN) ， 任 何 附加 的 安全 层 都 会 导 
致 应 用 变 慢 。 因 此 ， 在 保证 用 户 完成 意图 的 前 提 下 ， 你 需要 对 添加 的 安全 措施 (会 导致 延 
R) 进行 权衡 。 

本 章 将 探讨 应 用 安全 方面 的 一 些 重点 内 容 ， 而 不 会 做 深入 的 理论 研究 。 我 们 将 从 以 下 角度 
探讨 与 安全 性 相关 的 方法 。 











ini 





























注 1: 此 书 已 由 人 民 邮 电 








Er 








版 社 出 版 ，http:Wwww.ituring.com.cn/book/1068。 一 一 编者 注 
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。 应 用 访问 
如 何 访问 应 用 的 安全 、 如 何 管理 身份 ， 以 及 其 他 相关 的 主题 。 


























。 网 络 安全 

这 包括 与 服务 器 通信 相关 的 所 有 内 容 。 
。 本 地 存储 

与 设备 上 的 所 有 数据 相关 。 
。 数据 共享 





让 数据 从 其 他 应 用 进入 你 的 应 用 或 从 你 的 应 用 流向 其 他 应 用 。 


9.1 应 用 访问 

你 的 应 用 可 以 执行 验证 ， 也 可 以 不 执行 验证 。 对 于 大 多 数 游戏 、 新 闻 、 实 用 和 其 他 类 似 的 
应 用 而 言 ， 可 能 不 需要 身份 验证 。 

本 节 将 讨论 一 些 选择 ， 以 便 识别 设备 、 用 户 、 同 一 设备 上 的 多 个 用 户 ， 以 及 同一 或 多 个 设 
备 上 的 多 个 应 用 中 的 同一 用 户 。 

9.1.1 匿名 访问 

应 用 可 能 需要 验证 ， 也 可 能 不 需要 验证 。 例 如 ， 不 需要 订阅 的 新 闻 应 用 可 能 永远 都 不 需要 
































认证 。 然 而 ， 为 了 提供 个 性 化 新 闻 或 广告 ， 就 需要 为 设备 创建 唯一 的 标识 符 ， 比 如 Yahool 
Digest News 应 用 。 


有 两 个 选项 可 用 于 识别 设备 : 供应 商 的 标识 符 (Identifier for Vendor, IDFV) 和 广告 商 的 
标识 符 (Identifier for Advertiser，IDFA)。 下 面 我 们 将 详细 介绍 这 两 者 。 


IDFV (http://apple.co/1xxe8oK) 是 设备 上 每 个 应 用 的 持久 唯一 的 标识 符 ， 用 于 向 应 用 的 
供应 商标 识 设备 。 应 用 包 ID 的 一 部 分 用 于 生成 IDFV， 因 此 ， 即 使 应 用 来 自 同一 家 公司 ， 
IDFV 也 可 能 不 同 。 


使 用 -[UIDevice identifierForVendor] 方法 获取 IDFV。 如 果 用 户 在 设备 重启 后 没有 解锁 ， 
但 应 用 已 在 后 台 任 务 执行 期 间或 在 推送 通知 时 被 唤醒 了 ， 那 么 该 值 可 能 为 ntL。 如 果 是 这 样 
那么 就 稍 后 重 试 。 例 9-1 显示 了 检索 IDFV 的 一 些 简 单 代 码 。 不 要 在 主线 程 上 执行 此 操作 。 


例 9-1 检索 IDFV 


-(NSString *)idfv { 
UIDevice *device = [UIDevice currentDevice]; 
NSUUID *rv = device.identifierForVendor ; 
while(!rv) { 
[Thread sleepForTimeInterval:0.005]; 
rv = device.identifierForVendor ; 


















































} 


return rv.UUIDString; 


} 
Æ iOS 6, IDFV 是 从 包 ID 的 前 两 个 部 分 创建 的 。 因 此 ， 对 包 ID com.bundle.id.app1 而 
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言 ， 只 有 con. bundle 会 被 使 用 。 


iOS 7 中 存在 一 个 bug， 包 ID 27 9l com.bundle.id.appi 和 com. bundle.id.app2 的 两 个 应 
用 会 有 不 同 的 IDFV， 即 使 它们 来 自 于 同一 个 供应 商 (使 用 相同 证 书 来 签署 应 用 )。 荚 果 公 
司 并 未 修复 此 bug， 而 是 更 新 了 文档 。 

在 iOS 7 和 更 新 版 本 中 ， 除 了 最 后 一 部 分 ， 整 个 软件 包 ID 都 用 于 生成 IDFV。 因 此 ，IDFV 
的 生成 表 如 表 9-1 所 示 。 


表 9-1: 包 ID 的 部 分 内 容 用 于 生成 IDFV 


























包 ID iOS 6 iOS 7 及 更 新 版 本 
com.bundle.id.app1 com.bundle com.bundle.id 
com.bundle.id.app2 com.bundle com.bundle.id 
com.bundle.id.suite.appl com.bundle com.bundle.id.suite 
com.bundle.id.suite.app2 com.bundle com.bundle.id.suite 
simpleid? simpleid simpleid 


iOS 7 中 的 更 改 意 味 着 ， 现 在 你 可 以 使 用 两 个 选项 来 保留 唯一 的 设备 ID ， 并 跨 多 个 应 用 进 
行 跟踪 。 其 中 一 个 选项 是 ， 除 最 后 一 部 分 ， 保 持 包 ID 唯一 。 如 果 此 种 方式 不 可 行 ， 你 可 
以 选择 第 二 个 选项 一 一 使 用 共享 的 钥匙 串 来 共享 第 一 个 已 安装 应 用 所 获得 的 密 钥 。 

当 来 自 设备 且 必 于 同一 供应 商 的 所 有 应 用 都 被 卸载 时 ，IDFYV 会 被 重 置 。 因 此 ， 如 果 只 有 
一 个 应 用 ， 多 次 卸载 和 重新 安装 会 生成 不 同 的 ID. 
如 果 无 法 使 用 共享 钥匙 串 或 相同 的 最 后 一 部 分 包 ID， 那 么 你 将 无 法 在 多 个 应 用 之 间 唯 一 地 
识别 此 设备 。 

IDFA 是 可 重 置 的 标识 符 ， 在 设备 上 的 所 有 应 用 中 是 唯一 的 。 正 因为 在 众多 应 用 中 是 唯 
的 ， 所 以 它 才 是 真正 唯一 的 ID。 但 是 ，IDFA 可 以 被 用 户 重 置 。 此 外 ， 苹 果 公 司 对 它 的 
使 用 设置 了 限制 ， 你 必须 保证 在 提交 应 用 到 iTunes Connect 审核 时 使 用 它 。 此 ID 只 应 由 
广告 投放 系统 使 用 。 同 时 ， 它 还 带 有 一 个 表明 用 户 是 否 希望 使 用 此 ID 的 标签 。 根 据 文档 
(http://apple.co/LOyHGYa) 所 述 ， 如 果 标 签 未 启用 ， 那 么 IDFA 只 能 用 于 广告 的 频次 控制 、 
归属 、 转 换 事 件 、 估 计 唯 一 用 户 数 、 广 告 欺 诈 检 测 和 调试 。 

这 也 就 是 说 ， 你 可 以 用 IDFA 估算 应 用 的 唯一 用 户 数 ， 但 不 能 确定 特定 用 户 。 例 9-2 显示 了 检 
R IDFA 的 示例 代码 。 不 要 在 主线 程 上 调用 API， 因 为 返回 的 值 可 能 为 niL， 这 会 导致 重 试 。 
与 IDFV 一 样 ， 这 种 情况 是 有 可 能 发 生 的 ， 例 如 ， 如 果 设 备 已 重启 ， 但 用 户 尚未 解锁 设备 。 















































































































































例 9-2 检索 IDFA 
-(NSString *)idfa { 
ASIdentifierManager *mgr = [ASIdentifierManager sharedManager ]; 
if (mgr .isAdvertisingTrackingEnabled) { 
UUID *rv = mgr.advertisingIdentifier; 
while(!rv) { 
[Thread sleepForTimeInterval:0.005]; 





iE 2: 不 推荐 此 类 包 ID, 





A 
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9.1.2 
当 需 要 识别 用 户 时 ， 你 需要 认证 访问 。 这 并 不 意味 着 认证 必须 在 你 的 应 用 中 完成 。 以 下 是 





rv = mgr.advertisingIdentifier; 


} 
return rv.UUIDString; 


return nil; 


仍旧 使 用 UDID ? 


唯一 设备 标识 符 (unique device identifier, UDID) 现 已 弃 用 (JA iOS 6 FF 


始 )。 如 果 还 没 这 样 做 ， 你 应 该 删除 对 它 的 所 有 引用 。 





认证 访问 


一 些 可 用 的 认证 选项 。 
应 用 密码 


它 也 被 称 为 应 用 PIN， 无论 是 否 存在 登录 至 应 用 的 一 组 凭据 ， 应 用 PIN 都 是 你 想 要 添加 
到 应 用 的 本 地 凭据 。 实 际 上 ， 它 就 是 只 存储 在 设备 本 地 的 密码 。 例 如 ， 殴 用 管理 应 用 可 
能 永远 都 不 会 在 服务 器 上 存储 任何 数据 ， 但 仍 希 望 保护 在 设备 上 的 访问 。 男 一 方面 ， 医 


疗 记录 应 用 可 能 会 将 密码 作为 第 二 层 的 安全 保障 。 因 此 ， 用 户 首先 使 用 所 需 的 凭据 ( 通 








常 是 用 户 名 /电子 邮件 和 密码 ) 登录 ， 并 将 本 地 安全 作为 附加 层 添加 。 

















到 9-1 显示 了 两 个 应 用 ， 一 个 仅 使 用 了 本 地 凭据 ， 另 一 个 使 用 了 应 用 PIN 作为 辅助 安全 措施 。 
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图 9-1: 不 使 用 远程 凭证 的 Expense report 应 用 (£) 和 使 用 应 用 PIN 作为 第 二 层 安全 性 、 有 远 
42450) Credit Karma 应 用 (4) 





使 用 钥匙 串 在 本 地 存储 密码 。 不 要 在 文件 或 数据 库 中 以 未 加 密 的 形式 存储 。 








。 游戏 中 心 
此 选项 仅 适 用 于 游戏 。 用 GameKit 连接 游戏 中 心 ， 后 者 会 负责 使 用 凭据 对 用 户 进行 验 
证 。 游 戏 中 心 可 以 访问 用 户 资 料 、 个 人 记录 等 ， 但 仅 共 享 唯一 标识 用 户 所 需 的 内 容 CHII 
用 户 ID), 
例 9-3 显示 了 在 游戏 中 心 登录 后 获取 用 户 身份 的 模板 代码 。 


例 9-3 使 用 游戏 中 心 进行 登录 


#include «GameKit/GameKit.h» Qj 























(implementation HPLoginViewController 


-(void)authWithGameCenter { 
GKLocalPlayer _ weak *player = [GKLocalPlayer localPlayer]; @ 
if(!player.authenticated) ( © 
player.authenticateHandler - ^(UIViewController *vc, 
NSError *error) ( @ 
if(error) { 
// 处 理 错误 
} else if(vc) { @ 
[self presentViewController:vc animated:YES completion: { 


// 再 次 核实 用 户 现在 是 否 已 经 认证 








J 

} else { 
GKLocalPlayer *lp = player; 
if(lp) { 

[self verifyLocalPlayer:lp]; Q 

} 

} 

J; 
} else { 


[self verifyLocalPlayer:lp]; 
} 
} 


-(void)verifyLocalPlayer:(GKLoalPlayer *)player { 
[player generateIdentityVerificationSignatureWithCompletionHandler: 
^(NSURL *publicKeyURL, NSData *signature, 
NSData *salt, uint64_t timestamp, 
NSError *error) { @ 


if(error) { 
// 处 理 错误 ! 

} else { 
//player id = player.playerID 
// 使 用 数据 核实 © 





Hs 
} 


@end 


Q SIA GameKit 头 文件 。 不 要 忘记 与 Gamekit. framework 链接 。 

@ GkLocalPlayer 表示 运行 游戏 的 已 认证 玩家 。 在 任何 时 刻 ， 设 备 上 只 能 有 一 个 认证 玩家 。 

O 检查 玩家 是 否 通 过 认证 。 

O 设置 authenticateHandler 属性 将 触发 授权 。 

O 如 果 用 户 以 前 没有 授权 与 GameKit 连接 ， 那 么 在 回调 中 返回 的 视图 控制 器 必须 显示 
出 来 。 

Q 如 果 没 有 视图 控制 器 ， 也 没有 错误 ， 则 一 切 正常 。 但 是 需要 做 更 多 的 工作 以 获取 用 
户 的 详细 信息 。 

@ (E FA generatetIdentityVerificationSignatureWithCompletionHandler: 方法 获取 签 
名 ， 从 而 验证 本 地 玩家 。 

O 实际 的 验证 任务 应 该 发 生 在 服务 器 上 (https://developer.apple.com/library/ios/documen 
tation/GameKit/Reference/GKLocalPlayer_Ref/index.html#//apple_ref/occ/instm/GKLocal 
Player/generateldentity VerificationSignatureWithCompletionHandler:), ， 如 下 所 述 。 

















对 于 本 地 玩家 的 验证 〈 服 务 器 端 ) ， 请 遵循 下 列 步骤 。 
(1) 使 用 publicKeyURL 获取 X.509 HEB, URL 必须 是 域名 为 apple.com fy https URL. tk 





密 钥 必须 由 苹果 公司 签名 。 








(2) 按 顺 序 连 接 player .playerID、 应 用 的 包 ID、 大 字 节 UInt 64 格式 的 timestamp 和 Salt, 
(3) 生成 连接 数据 的 SHA-1 散 列 。 

(4) 用 步骤 1 中 下 载 的 公 钥 验证 针对 此 散 列 的 签名 。 

(5) 如 果 它 们 匹配 ， 则 一 切 正常 。 用 户 进 行 了 验证 ， 且 player .playerID 也 可 以 使 用 。 





第 三 方 认 证 
语义 与 游戏 中 心 的 身份 验证 类 似 ， 因 为 你 拥有 用 户 和 登录 体验 。 有 具体 的 SDK 不 在 这 里 
袭 述 ， 但 你 可 以 随时 查阅 Facebook (https://developers.facebook.com/docs/ios), 、Google+ 
(http://bit.ly/gp-signin) 或 Twitter (https://dev.twitter.com/twitter-kit/ios/twitter-login) 的 第 
三 方 认证 API. 


你 自己 的 验证 
大 多 数 应 用 选择 保留 对 注册 和 登录 过 程 的 完全 控制 ， 这 需要 自 定义 的 认证 机 制 。 使 用 具 
有 密码 的 电子 邮件 / 用 户 名 作为 凭证 是 最 常用 的 认证 机 制 。 我 们 简要 讨论 一 下 在 实施 这 
一 过 程 中 要 采取 的 关键 措施 。 
4 强制 要 求 使 用 强 密码 ， 长 度 至 少 为 六 个 字符 ， 包 含 大 小 写字 符 (如 果 适 用 ， 可 以 使 
用 罗马 字母 表 )、 数 字 和 特殊 字符 。 
一 些 应 用 设置 了 最 大 长 度 的 限制 ， 但 这 一 措施 不 是 特别 好 。 这 好 像 在 对 用 户 说 ， 
“Wie! 很 抱歉 ， 我 们 不 允许 有 更 高 的 安全 性 ”。 










































































此 外 ， 另 一 个 重点 是 ， 较 长 但 易于 记忆 的 密码 可 能 会 比较 短 且 模糊 的 密码 更 难 破解 。 
然而 、 因 为 在 移动 设备 上 给 入 较 长 的 密码 是 比较 繁 开 的 ， 所 以 通常 选择 较 短 和 较 复 
杂 的 密码 。 


4 提供 活动 会 话 的 列表 ， 并 允许 用 户 将 其 他 设备 或 位 置 上 的 现 有 会 话 设 为 无 效 。 

e 支持 双 因 素 身 份 验证 ， 并 在 遇 到 异常 行为 时 使 用 。 例 如 ， 在 异常 时 间 从 新 位 置 或 从 
新 设备 登录 。 

4 就 与 金融 或 金钱 相关 的 应 用 而 言 ， 启 用 会 话 超时 机 制 〈 例 如 ， 如 果 应 用 在 后 台 停 留 
了 特定 时 间 , 如 超过 5 或 10 分 钟 , 则 将 会 话 设 为 无 效 )。 这 与 启用 网 站 上 的 “ 记 住 我 ” 
选项 类 似 。 

4 或 者 ， 在 用 户 使 用 非 到 期 访问 令 牌 永久 登录 时 ， 使 用 较 短 的 应 用 PIN 进行 本 地 身份 
验证 。 图 9-1 展示 了 带 有 本 地 应 用 PIN 的 Credit Karma 应 用 。 

e 对 于 永久 登录 ， 确 保 访 问 令 牌 (类 似 于 浏览 器 中 的 Cookie) 存储 在 本 地 的 钥匙 串 

+ 启用 CAPTCHA (你 可 以 用 限制 性 的 方式 使 用 此 功能 ， 例 如 ， 你 可 以 在 连续 3 或 5 
次 的 无 效 尝试 后 启用 )。 

e 或 者 ， 使 用 本 地 身份 验证 将 Touch ID 与 钥匙 串 集 成 ， 从 而 进行 无 密码 登录 。 

e 请 遵循 9.2.2 节 和 9.3 节 中 讨论 的 最 佳 实践 。 


使 用 铀 是 串 或 Touch ID 这 样 的 加 密 选项 会 增加 开销 ， 从 而 导致 对 用 户 的 响应 
变 慢 。 

即使 使 用 最 新 的 更 新 版 本 ，Touch ID 也 是 缓慢 且 不 可 靠 的 ， 特 别 是 可 能 出 现 
指纹 不 被 识别 ， 导 致 多 次 重 试 。 


9.2 网络 安全 


第 7 章 对 网 络 进行 了 深入 的 讨论 。 本 节 将 讨论 在 与 远程 设备 通信 中 与 安全 有 关 的 最 佳 实 
践 ， 该 远程 设备 可 以 是 服务 器 ， 也 可 以 是 点 对 点 设备 。 






























































9.2.1 使 用 HTTPS 


假设 你 将 HTTP 作为 底层 消息 传递 协议 (TCP 是 传输 层 协议 ) ， 那 么 你 必须 通过 TLS/SSL 
使 用 它 。 这 也 就 是 说 ， 你 应 该 一 直 使 用 HTTPS。 但 是 ， 使 用 HTTPS 有 几 个 问题 。 如 果 这 
些 潜在 风险 未 得 到 解决 ， 则 HTTPS 可 能 会 受到 影响 。 

1. CRIMEA} 

不 要 使 用 SSL/TLS 压缩 。 如 果 你 现在 在 使 用 ， 请 在 继续 之 前 立即 关闭 它 。 这 会 让 你 处 于 较 
大 的 风险 当中 。 使 用 TLS 压缩 (gzip, deflate 或 其 他 格式 )， 任 何 请 求 都 会 受到 CRIME 
(Compression Ratio Info-leak Made Easy， 压 缩 率 使 信息 很 容易 泄露 ) 攻击 。 要 想 缓解 风险 ， 
可 以 关闭 TLS 压缩 ， 并 给 每 个 响应 发 送 反 CRIME cookie， 较 为 简单 的 方式 是 发 送 一 个 唯 
一 的 随机 序列 cookie。 

2. BREACH 攻 击 

如 果 使 用 请 求 / 响 应 正文 压缩 (Transfer-Encoding = gzip 或 defLate)， 你 的 通信 会 受到 
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BREACH (Browser Reconnaissance and Exfiltration via Adaptive Compression of Hypertext, 
通过 自 适应 超 文本 压缩 的 浏览 器 侦 听 和 渗透 ) 攻击 ， 这 种 攻击 类 型 于 2012 年 9 月 首次 发 
现 。 当 满足 以 下 标准 时 ， 就 会 存在 风险 。 

。 应 用 使 用 HTTP 压缩 。 

。 响应 反映 了 用 户 输入 。 

。 响应 反映 了 隐私 。 


没有 单一 的 方法 可 以 降低 这 种 风险 。The Breach Attack (http://www.breachattack.com) 网 站 

按 有 效 性 列 出 了 以 下 方法 。 

。 禁用 HTTP 压缩 。 这 种 方法 增加 了 传输 的 数据 量 ， 可 能 不 会 作为 实际 的 解决 方案 。 

。 从 用 户 输入 分 离 出 隐私 。 将 授权 码 放 在 远离 请 求 正文 的 地 方 。 

。 对 每 个 请 求 进行 随机 化 加 密 。 但 是 ， 由 于 每 个 请 求 的 加 密 是 随机 的 ， 因 此 ， 多 个 并 行 请 
求 可 能 无 法 实现 了 。 

。 修饰 隐私 。 不 要 以 原始 格式 发 送 隐 私 。 

。 使 用 CSRF 保护 易 受 攻击 的 HTML 页 面 。 在 移动 原生 应 用 上 ， 除 非 使 用 移动 Web， 否 
则 不 需要 CSRF。 

。 隐藏 长 度 。 一 个 较 好 的 方法 是 在 HTTP 响应 中 使 用 分 块 传输 编码 。 

。 对 请 求 限 速 (这 应 该 作为 最 后 的 方法 )。 


9.2.2 ”使 用 证 书 锁定 


HTTPS 不 是 万 灵 药 一 一 采取 HTTPS 不 会 神奇 地 确保 所 有 的 通信 都 是 安全 的 。HTTPS 的 
基础 是 对 公 钥 的 信任 ,该 公 钥 用 于 加 密 初 始 消息 (在 SSL 握手 期 间 )。 中 间 人 (man-in-the- 
middle, MITM) 攻击 会 捕获 用 于 加 密 消息 的 密 钥 。 


图 9-2 显示 了 中 间 人 攻击 的 概要 ， 其 中 中 介 器 (如 设备 连接 的 WiFi 热点 或 正在 使 用 的 代理 
服务 器 ) 拦截 来 自 设 备 的 请 求 。 当 设备 发 送 对 服务 器 证 书 的 请 求 时 ， 中 介 器 将 请 求 发 送 到 
服务 器 并 捕获 其 应 答 。 然 后 ， 它 并 没有 将 密 钥 返回 至 设备 ， 而 是 返回 了 自己 的 密 钥 。 这 与 
Charles 代理 服务 器 使 用 的 技术 相同 (参见 7.3.3 节 )。 





















































































gue MS 获 取 公 共 密 包 
TE — 捕获 服务 器 密 旬 
发 送 黑 客 的 密 铀 生成 黑客 的 密 铀 














9-2: PAARE 


不 让 请 求 变 成 无 效 的 唯一 方法 就 是 信任 ， 该 信任 由 网 络 库 放 置 在 接收 到 的 证 书 之 中 。 证 书 
只 是 签名 的 公 钥 。 因 此 ， 如 果 网 络 库 信 任 签名 者 ， 那 它 也 会 信任 主机 提供 的 公 钥 。 黑 客 提 
供 的 假 的 根 证 书 成 为 了 让 所 有 安全 措施 崩溃 的 罪魁 祸首 。 





























找到 用 于 开发 并 安装 了 Charles 证 书 的 个 人 设备 并 不 难 。 况 且 ， 因 为 私 钥 和 
证 书 在 公共 域 中 都 是 可 用 的 ， 所 以 使 用 证 书 作 为 攻击 的 起 点 并 不 罕见 。 
市 场 上 的 二 手 设备 有 多 少 真 的 可 以 信任 ? 越狱 一 台 iOS 设备 是 相当 容易 的 。 
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个 问题 的 解决 方案 就 是 所 谓 的 证 书 锁定 (https:/www.owasp.org/index.php/Certificate_ 
it Public_Key_Pinning)。 这 种 方案 的 工作 原理 是 ， 通 过 只 信任 一 个 或 儿 个 能 够 作为 应 用 
根 证 书 的 证 书 ， 应 用 创建 一 个 自 定义 的 信任 级 别 。 这 允许 应 用 仅 信 任 来 自白 名 单 的 证 书 ， 
确保 设备 上 永 不 安装 那些 允许 网 络 监视 的 未 知 证 书 。 


在 使 用 NsURLConnection 时 ， 你 可 以 提供 一 个 执行 证 书 验证 的 NSURLConnection Delegate, 
例 9-4 显示 了 可 在 应 用 中 实现 证 书 锁定 的 代表 性 代码 ? 
例 9-4 ”证书 锁定 


typedef void(^HPResponseHandler)(NSURLResponse *, NSError *error); 












































@interface HPPinnedRequestExecutor @ 


@property (nonatomic, readonly) NSURLRequest *request; 
@property (nonatomic, copy) HPResponseHandler handler; 


@end 

@interface HPPinnedRequestExecutor () <NSURLConnectionDelegate> 

@property (nonatomic, readwrite) NSURLRequest *request; 

@end 

(implementation HPPinnedRequestExecutor 

-(instancetype)initWithRequest:(NSURLRequest *)request { 
if(self = [super init]) { 


self.request - request; 


j 


return self; 


-(void)executeWithHandler:(HPResponseHandler)handler { 
self.handler - handler; 
[[NSURLConnection alloc] initWithRequest:self.request delegate:self]; 


- (void)connection:(NSURLConnection *)connection 
didReceiveResponse:(NSURLResponse *)response { 


// 做 常规 的 事情 ,用 处 理 器 发 送 结果 





























注 3: 改编 自 OWASP (https://www.owasp.org/index.php/Certificate_and_Public_Key_Pinning#iOS) 。 
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-(BOOL)connection:(NSURLConnection *)connection 
canAuthenticateAgainstProtectionSpace: (NSURLProtectionSpace*)space { 
return [NSURLAuthenticationMethodServerTrust 

isEqualToString:space.authenticationMethod]; @ 


} 


- (void)connection:(NSURLConnection *)connection 
didReceiveAuthenticationChallenge: (NSURLAuthenticationChallenge *)challenge {© 


void (^cancel)() = ^( 
[challenge.sender cancelAuthenticationChallenge: challenge]; 


Jo 


if ([NSURLAuthenticationMethodServerTrust 
isEqualToString:challenge.protectionSpace.authenticationMethod]) { 


SecTrustRef serverTrust = challenge.protectionSpace.serverTrust; 


if(serverTrust == nil) { 
cancel(); 
return; 
} 
OSStatus status = SecTrustEvaluate(serverTrust, NULL); 
if(status !- errSecSuccess) { 
cancel(); 
return; 
} 


SecCertificateRef svrCert = SecTrustGetCertificateAtIndex(serverTrust, 0); 
if(svrCert == nil) { 

cancel(); 

return; 


j 


CFDataRef svrCertData - SecCertificateCopyData(svrCert); 
if(svrCertData == nil) { 

cancel(); 

return; 


j 


const UInt8* const data - CFDataGetBytePtr(svrCertData); 
const CFIndex size = CFDataGetLength(serverCertificateData) ; 
NSData* certi = [NSData dataWithBytes:data length:(NSUInteger)size]; 


if(certi1 == nil) { 
cancel(); 
return; 


j 


NSString *file - [[NSBundle mainBundle] 
pathForResource:@"pinned-key" 
ofType:@"der"]; 

NSData* cert2 = [NSData dataWithContentsOfFile: file]; 


if(cert2 == nil) { 
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cancel(); 
return; 


j 


if(![certi isEqualToData:cert2]) { 
cancel(); 
return; 


}@ 


[challenge. sender 
useCredential:[NSURLCredential credentialForTrust:serverTrust] 
forAuthenticationChallenge:challenge]; @ 


} 
@end 


@ 实现 NsURLConnectionDelegate 协议 。 

@ connection:canAuthenticateAgainstProtectionSpace: 方法 检查 委托 是 否 能 够 响应 保护 
空间 的 身份 验证 形式 。 对 于 SSL (服务 器 信任 ) ， 返 回 YES, 

@ connection:didReceiveAuthenticationChallenge: 方法 处 理 chaLLenge， 可 以 取消 认证 
(无 效 时 ) 或 使 用 凭证 (有 效 时 )。 

OQ 如 果 发 生 了 失败 ， 则 将 证 书 置 为 无 效 ， 与 找 不 到 证 书 或 与 绑 定 的 密 钥 不 匹配 时 一 样 。 

Q 如 果 一 切 都 成 功 了 ， 将 证 书 置 为 可 用 。 








m 











iOS 8 及 更 新 版 本 的 注释 
委托 回调 方法 connection:canAuthenticateAgainstProtectionSpace: 和 connection:di 


dReceiveAuthenticationChallenge: 在 iOS 8 中 被 齐 用 ， 改 为 connection:willSendRequ 
estForAuthenticationChallenge: 17, 


根据 验证 结果 ， 委 托 需 要 调用 NsSURLAuthenticationChallengeSender 协议 的 以 下 方法 
之 一 (使 用 challenge.sender 对 象 ) 。 


e useCredential:forAuthenticationChallenge: 
验证 成 功 。 使 用 证 书 。 

e cancelAuthenticationChallenge: 
验证 失败 。 取 消 请 求 。 

e continueWithoutCredentialForAuthenticationChallenge: 
没有 证 书 也 继续 (不 要 这 样 做 )。 

e performDefaultHandlingForAuthenticationChallenge: 
让 请 求 通过 系统 提供 的 默认 路 由 进行 。 

。 rejectProtectionSpaceAndContinueWithChallenge: 


拒绝 当前 提供 的 保护 空间 。 这 种 情况 比较 少见 ， 如 果 有 的 话 , 则 用 于 SSL 证 书 验证 。 





T 
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类 似 的 方法 可 用 于 实现 NSURLSession 对 象 。 这 允许 在 会 话 的 范围 内 进行 控制 ， 而 不 必 担 
心 应 用 创建 的 每 个 请 求 。 


尔 不 必 维 护 大 量 的 代码 。 像 RNPinned CertValidator (https://github.com/ 
rnapier/RNPinnedCertValidator) 这 样 的 库 有 助 于 将 代码 减少 到 几 行 (http:// 
robnapier.net/pinning-your-ssl-certs ) 。 

如 果 使 用 AFNetworking 库 ， 你 可 能 需要 编写 不 同 的 代码 。 互 联网 上 的 一 些 
教程 ”可 以 为 你 提供 一 些 帮助 。 


9.3 本 地 存储 


与 通过 网 络 交换 的 数据 类 似 ， 存 储 在 设备 上 的 数据 是 不 能 防止 被 自 改 的 ， 而 且 如 果 不 小 心 
处 理 的 话 ， 入 侵 者 是 可 以 读 取 或 修改 数据 的 。 以 下 是 需要 注意 的 几 个 要 点 ， 以 及 为 了 保护 
本 地 存储 空间 需要 遵循 的 最 佳 实践 。 


。 本 地 存储 不 安全 

在 越狱 设备 上 非常 容易 访问 本 地 存储 。 如 果 观 看 了 本 章 开 头 引 用 的 视频 ， 你 可 能 会 注意 
到 ， 使 用 一 些 即 播 即 用 的 工具 就 可 以 替换 或 修改 这 些 文件 。 
你 可 能 会 说 ,“ 这 只 是 众多 设备 中 的 一 个 ， 并 且 自 改 数据 也 是 为 了 设备 上 的 用 户 "。 我 同 
意 这 种 说 法 。 但 了 解 此 类 型 的 算 改 对 整个 应 用 生态 系统 产生 的 副作用 却 是 非常 重要 的 。 
例如 ， 在 邮件 应 用 中 ， 设 备 可 能 会 被 注入 用 于 发 送 邮 件 的 一 些 数据 ， 这 会 导致 应 用 很 容 
易 就 发 送 大 量 邮件 。 即 使 用 户 之 后 会 被 列 入 黑 名 单 或 被 阻止 ， 但 伤害 已 经 造成 了 。 服 务 
器 应 实现 额外 的 安全 性 ， 使 用 速率 限制 技术 和 增强 的 DDoS 保护 措施 来 进行 防卫 。 


。 加 密 本 地 存储 
本 地 存储 可 以 利用 操作 系统 提供 的 数据 保护 能 力 进行 加 密 。 
要 想 启 用 数据 保护 ， 打 开 Xcode， 然 后 选择 目标 。 在 “功能 ”选项 下 ， 查 找 “ 数 据 保 
护 ”， 并 将 其 打开 ( 见 图 9-3)。 这 将 为 应 用 ID 添加 数据 保护 权利 。 





















































v a Data Protection Cy ) 











Steps: v Add the "Data Protection" entitlement to your App ID 











9-3; 在 Xcode 中 启用 数据 保护 





注 4: iOS Developer Library, "Authentication Challenges and TLS Chain Validation”(http:/apple.co/1EIV6Ud). 
ik 5: 例如 ，Eric Allam [fJ “AFNetworking SSL Pinning with Self-Signed Certificates" (http://initwithfunk.com/ 
blog/2014/03/12/afnetworking-ssl-pinning-with-self-signed-certificates/) 。 








在 默认 情况 下 ， 启 用 数据 保护 后 ， 应 用 使 用 的 所 有 本 地 存储 都 将 使 用 设备 密码 进行 加 
密 。 这 意味 着 在 设备 解锁 前 无 法 访问 数据 。 
你 可 以 在 Apple Developer 门户 上 配置 安全 级 别 ， 导 航 至 Certificates, Identifiers & 


Profiles 一 Identifiers。 转 到 应 用 ID 子 部 分 ， 然 后 选择 要 配置 的 应 用 ID。 你 应 该 能 注意 
到 数据 保护 是 启动 状态 〈 见 图 9-4)。 


High Performance iOS Apps com.m10v.hperf 


Name: High Performance iOS Apps 


ID Prefix: 


ID: com.m10v.hperf 











Application Services: 


Service Development Distribution 
App Group @ Enabled @ Enabled 
Associated Domains @ Enabled @ Enabled 
Data Protection @ Enabled @ Enabled 
Game Center @ Enabled @ Enabled 
HealthKit © Disabled © Disabled 
Homekit © Disabled © Disabled 


Wireless Accessory 


E © Disabled © Disabled 
Configuration 
iCloud 上 Enabled @ Enabled 
In-App Purchase ® Enabled @ Enabled 
Inter-App Audio © Disabled © Disabled 











9-4; Apple Developer — App Capabilities Con guration 一 Data Protection 


单 击 编辑 按钮 即 可 配置 功能 。 图 9-5 展示 了 数据 保护 功能 的 安全 级 别 。 








Data Protection 
€ Enabled 


Sharing and Permissions 
© Complete Protection 
Protected Unless Open 


Protected Until First User Authentication 











图 9-5. 数据 保护 的 安全 级 别 
共享 和 权限 选项 如 下 。 


。 完全 保护 : 在 任何 时 候 访问 文件 进行 读 取 或 写 人 人 时， 设备 都 需要 是 解锁 状态 的 。 设 备 锁 
定 后 不 久 (如 果 “ 需 要 密码 ”一 项 设置 为 “立即 ”时 ， 则 在 10 秒 后 ) ， 加 密 密 钥 会 被 废 
弃 ， 导 致 所 有 数据 无 法 访问 ， 直 到 设备 再 次 解锁 。 
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打开 前 保护 ;这 需要 在 创建 文件 句柄 时 解锁 设备 ,一旦 句柄 可 用 ， 则 设备 锁定 时 也 可 以 
写 入 内 容 。 这 一 选项 在 设备 已 经 锁定 ， 但 还 需要 对 文件 进行 写 入 时 比较 有 用 。 例 如 ， 当 
设备 已 经 被 解锁 ， 用 户 可 以 触发 邮件 应 用 来 下 载 附件 ， 而 在 设备 被 锁定 之 后 ， 应 用 可 以 
触发 后 台 操作 继续 下 载 。 

第 一 个 用 户 认 证 前 保护 : 这 需要 设备 在 重新 启动 后 进行 一 次 解锁 。 第 一 次 解锁 后 ， 应 用 
可 以 访问 所 有 文件 ， 并 且 没 有 任何 限制 。 

需要 注意 的 是 ， 无 论 安全 级 别 是 多 少 ， 一旦 启用 数据 保护 ， 重 新 启动 后 不 能 立即 访问 文 
件 一 一 用 户 必须 至 少将 设备 解锁 一 次 。 这 也 意味 着 ， 应 用 在 接收 推送 通知 后 尝试 读 / 写 
文件 时 ， 如 果 设 备 在 重新 启动 后 从 未 解锁 ， 那 么 将 导致 错误 。 

对 于 每 个 文件 的 保护 ， 你 可 以 使 用 方法 - [NSData writeToFile:options:error:]; 对 
于 完全 保护 ， 将 选项 设置 为 NSDataWritingFileProtectionComplete ; 对 于 打开 前 保护 ， 
iz BY NSDataWritingFileProtectionCompleteUnlessOpen; 对 于 第 一 个 用 户 认 证 前 保 
护 ， 设 置 为 NSDataWritingFileProtectionCompleteUntilFirstUserAuthentication。 注 
意 ， 如 果 在 应 用 级 别 启用 了 数据 保护 ， 则 默认 值 与 开发 者 门户 上 设置 的 一 致 。 

或 者 ， 你 可 以 使 用 方法 - [NSFileManager createFileAtPath:contents:attributes:], 
将 属性 字典 设置 为 @{NSFileProtectionKey:<required-level>} (如 果 需 要 ， 还 有 其 他 属 
性 )， 其 中 required-level 可 以 是 NSFileProtectionComplete, NSFileProtectionComple 
teUnlessOpen 或 NSFileProtectionCompleteUntilFirstUserAuthentication, 


此 外 ， 使 用 属性 - [UIApplication protectedDataAvailable] 确定 是 否 可 以 访问 受 保护 
的 文件 。 如 果 设 备 未 锁定 或 未 启用 数据 保护 ， 则 该 值 会 设置 为 YYS。 当 修改 属性 值 为 NO 
It, LAW NSFileProtectionComplete 或 NSFileProtectionCompleteUnlessOpen 属性 的 文 
件 无 法 被 访问 ， 直 到 设备 解锁 ， 同时， 具有 NSDataWritingFileProtectionCompleteUnti 
lFirstUserAuthentication 属性 的 文件 也 无 法 被 访问 ， 直 到 设备 重新 启动 ， 并 解锁 。 
NSUserDefaults 不 安全 

通常 来 说 ， 我 们 总 是 认为 用 户 默 认 值 被 保存 在 安全 的 地 方 。 事 实 上 ， 它 们 很 简单 。plist 
文件 是 与 其 他 应 用 文件 一 起 保存 的 。” 

NSBundle 值 也 不 安全 

噢 ! 应 用 包 的 设置 被 当 作 是 与 应 用 捆绑 在 一 起 的 ， 从 不 修改 。 这 种 理解 只 有 部 分 是 正确 
的 ， 因 为 包含 值 的 .plist 文件 实际 上 是 可 以 被 自 改 的 。 

不 要 完全 依靠 钥匙 串 

钥匙 串 的 安全 可 以 被 破坏 。 攻 击 者 在 设备 锁定 时 肯定 无 法 访问 关键 信息 。 然 而 ， 重 要 的 
是 不 要 过 分 依赖 钥匙 串 。 因 为 加 密 密 钥 是 通过 使 用 仅 为 4 位 数字 的 设备 密码 ， 经 过 预 
定 公式 产生 的 。 所 以 这 仅 提 供 了 最 多 10 000 种 组 合 一 一 大 家 都 知道 ， 当 涉及 安全 性 时 ， 
10 000 种 组 合 根本 不 算是 强大 的 安全 防护 。 考 虑 到 iOS 设备 在 6 次 错误 尝试 后 ， 用 户 在 
一 分 钟 内 不 允许 进行 登录 (https;//support.apple.com/en-us/HT204306) ， 因 此 ，10 000 种 
组 合 可 以 在 几 个 小 时 内 手动 完成 。 









































































































































E 6: Prateek Gianchandani, InfoSec Institute, “iOS Application Security Part 20 - Local Data Storage (NSUser 





Defaults, CoreData, Sqlite, Plist Files)" (http://resources.infosecinstitute.com/ios-application-security-part- 


20-local-data-storage-nsuserdefaults-coredata-sqlite-plist-files/). 
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但 是 ， 一 般 来 说 ， 攻 击 者 不 必 花 费 那 么 多 时 间 。 根 据 几 年 前 发 布 的 一 份 报告 (http:/ 
danielamitay.com/blog/201 1/6/13/most-common-iphone-passcodes) ， 前 10 个 密码 的 使 用 量 
占据 了 当前 正在 使 用 的 密码 总 数 的 15% 。 

好 在 最 近 的 iOS 版 本 有 一 个 可 以 启用 更 强 密码 的 选项 。 你 可 以 进入 Settings ^ Touch ID 
& Passcode， 然 后 你 会 发 现 Simple Passcode 选项 是 被 选中 的 ( 见 图 9-6)。 








< Settings Touch ID & Passcode 


FINGERPRINTS 


Add a Fingerprint... 


Turn Passcode Off 


Change Passcode 


Require Passcode Immediately 


Simple Passcode WU 


A simple passcode is a 4 digit number. 


Voice Dial 


3 


Music Voice Control is always enabled. 


ALLOW ACCESS WHEN LOCKED: 


Today 
Notifications View 


Passbook 


@ 9 @ 4 


Reply with Message 











9-6. 简单 密码 


简单 密码 是 由 4 位 数字 组 成 的 ， 而 非 简单 密码 可 以 是 任意 长 度 的 ， 并 且 可 以 包括 字母 、 
数字 、 字 符 以 及 特殊 字符 。 如 图 9-7 所 示 ， 每 个 选项 的 键盘 都 不 同 。 虽 然 使 用 字母 数字 
密码 是 一 个 不 错 的 选择 ， 但 启用 此 选项 的 用 户 总 数 非常 少 ， 所 以 不 能 依赖 它 。 

作为 一 种 可 行 的 做 法 ， 你 可 以 对 存储 在 密 钥 库 中 的 数据 进行 加 密 ， 并 只 存储 最 少 的 数 
据 。 还 可 以 针对 每 个 设备 生成 密 钥 ， 并 在 本 地 存储 。 再 次 强调 ， 最 好 的 做 法 是 让 攻击 
者 更 加 难以 定位 和 解密 数据 。 如 果 攻 击 者 可 以 物理 访问 设备 ， 那 你 能 做 的 也 就 只 有 这 
么 多 了 。 
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Cancel Change Passcode Cancel Change Passcode 
Enter your new passcode Enter your new passcode 
1 2 3 
£ 3 Qa w ERIT YUI!!O 
4 5 6 
ai M ub AIlSIDIFIGIHIJIKIL 
7 8 9 ry | 
PARS TUV WXYZ t ZXCV BNM | 
0 a v Q space return 




















9-7. 简单 的 密码 使 用 4 位 数字 ( 左 ) ， 而 复杂 的 密码 可 以 使 用 字母 数字 和 特殊 字符 (6) 


小 心 记 录 的 东西 
使 用 内 置 的 NSLog ER 


函数 进行 日 志 记录 是 很 常见 的 ， 因 为 这 是 开发 人 员 被 教导 的 方式 。” 


方 文档 (https://developer.apple.com/library/ios/documentation/Cocoa/Reference/Foundation/ 





Miscellaneous/Foundation_Functions/#//apple_ref/c/func/NSLog ) 

















果 系 统 日 志 工具 ”的 函数 。 
至 可 以 在 记录 的 很 多 天 后 看 到 这 


在 Xcode 中 ， 导 航 到 Window 
9-8 所 示 。 








文 些 日 志 ^o 


一 Devices， 你 应 该 能 看 到 连 





车 接 设 备 和 模拟 


声明 了 “将 错误 消息 记录 到 芋 
它 不 是 控制 台 日 志 记录 。 而 且 也 没有 iOS 设备 控制 台 


。 你 其 


以 器 列表 ， 如 图 














e 
Device Information 
DEVICES — 
m My Mac Name iPhone 2 | e= 
104 10 (144389) Model iPhone 6 Plus | 
Phone Capacity 55.49 GB (43.84 GB available) 
8. 2 (120509) 
Battery 9596 
SIMULATORS ios 8.2 (120508) 
iPad 2 M > ^ ^ 
Ul 716506» Identifier E; 
iPad 2 \ O | 
Ul 6109» View Device Logs | Take Screenshot — 
iPad 2 
Wl 5202099 
Installed Aj 
m iPad Air in om 
wa 7.1 (110167) 
wa iPad Air È HPerf Apps 2 com.m10v.hperf 
8.1 (128411 E 
) t " 35214wildcard a 
iPad Air z 
Wl 62020509) ewe 343 
3 Loe Retina 
11D167) * 
9-8: 设备 概览 





E7: 从 技术 上 讲 ， 这 不 是 数据 存储 。 但 设备 日 志 类 似 于 数据 存储 ， 














具有 自动 和 永久 的 特点 。 








Ba 


x 
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如 果 单 击 View Device Logs 按钮 ， 你 应 该 可 以 看 到 所 有 的 日 志 ， 如 图 9-9 所 示 。 





NSLog 的 输出 。 











这 包括 





























a 
" Incident Identifier: ^ Ee ESRAS g 
MOSA A! Logs CrashReporter Key: wma "ic8db0017 140007. s 
Process^ Type Date/Time Hardware Model: EATUR 1 
: = Process: Spotify [1371 
Spotify ^ Crash 3/14/15, 7:39 PM Path: 7private/var/mobile/Containers/Bundle/Application/^ 
Spotify ^ Crash 3/14/15, 7:39 PM ^-4178-81E3-^ Aut gU/Spot ify, app/Spotify 
Spotify — Crash 3/14/15, 7:38 PM || Identifier: cor cpa ify client 
Spotify ^ Crash Code Type: ARM-64 (Native) 
Unknown Unknown — 3/29/15, 9:08 AM || Parent Process: Vaunchd 12) 
Unknown Unknown 3/29/15, 7:12 AM Date/Tine: 2015-03-14 19:38:10.947 -0700 
j Launch Time: 15-03-14 19:37:49.264 -07 
Unknown Unknown 3/28/15, 8:06 AM Os’ Versión: 105 8.2 (12D508) 
Unknown Unknown 3/27/15, 7:57 PM Report Version: 105 
Unk Unk 3/27/15, 7:29 PM : 
DEW: Own : Exception Type: 00000020 
Unknown Unknown 3/26/15, 5:49 PM Exception Codes: 0x000000008badfü0d 
Unknown Unknown 3/26/15, 12:48 PM Highlighted Thread: 5 
Unknown Unknown 3/26/15, 6:58 AM Application Specific Information: 
Unknown Unknown 3/24/15, 11:04PM com.spotify.client failed to scene-update in time 
Unknown Unknown ^ 3/24/15,10:59PM Elapsed total CPU time (seconds): 0.620 (user 0.620, system 0.000), 30% CPU 
Unknown Unknown 3/24/15, 6:54 PM Elapsed application CPU time (seconds): 0.002, 0s CPU 
Unknown Unknown 3/28/15, 11:28 PM Thread 0 name: Dispatch queue: com.apple.main-thread 
Unknown Unknown 3/23/15, 11:28 PM Thread 0: 
a 0 libsystem_kernel.dylib Ox0000000193bdf0c0 __psynch_mutexwait + 8 
Unknown Unknown 3/23/15,11:28PM| | 1 {ibsysten pthread. dylib 0x0000000193c79490 _pthread_mutex_lock + 416 
Unknown Unknown 3/23/15, 9:57 AM 2 Spotify 0x0000000100d4226c 0x10001c000 + 13787756 
Unknown Unknown  3/22/15,12:42AM 3 Spotify 0x0000000100d48714 0x10001c000 + 13813524 
4 4 Foundation 0x0000000183363510 _ NSFireTimer + 88 
Unknown Unknown 3/22/15, 12:42AM _ 5  CoreFoundation 0x0000000182422c1c 
Unknown Unknown 3/21/15,3:48 PM  . CFRUNLOOP IS CALLING OUT. TO A TIMER CALLBACK FUNCTION + 24 
Se 6  CoreFoundation 0x00000001824228cc . CFRunLoopDoTimer + 884 
Unknown Unknown 3/21/15, 2:50 PM 7  CoreFoundation 0x0000000182420318 . CFRunLoopRun + 1368 
Unknown Unknown 3/21/15, 2:50 PM s coreroundatton 050000800192340 ra CERun oc pu peed THe + 392 
" raphicsServices xi ventRunModal + 
Unknown Uniown “SUIS ROAM: | | 6: aur 0x0000000186cde108 UlApplicationMain + 1484 
Unknown Unknown 3/20/15, 2:24 PM 11 Spotify — 0x00000001000855ac 0x10001c000 + 431532 
Unknown Unknown 3/19/15, 6:16 PM 12 libdyld.dylib 0x0000000193ac6a04 start + 0 
Unknown Unknown 3/18/15, 11:23 PM Thread 1 name: Dispatch queue: con.apple. Libdispatch-nanager 
3 Thread 1: 
Unknown. Unknown: s/TS/IS TES CAM [IN ot CSS rice. Lame ditto 0x0000000193bc4c24 kevent64 + 8 
Unknown Unknown 3/18/15, 7:13 AM 1  libdispatch.dylib 0x0000000193aa9e6c dispatch mgr invoke + 272 
Done 
MY - 
图 9-9: 设备 日 志 


作为 最 佳 实践 ， 不 要 在 非 调试 版 本 中 使 用 NSLog。 一 个 较 好 的 方法 是 使 用 包装 函数 和 


如 例 9-5 所 示 。 








1.4.4 节 中 深入 讨论 了 日 志 记录 。 


fil 9-5 使 用 NSLog 记录 日 志 


(implementation HPLogger 


*(void)log:(NSString *)format, 


#ifdef DEBUG @ 
va_list args; 


va_start(args, format); 


NSLogv( format, 
va_end(args); 
#endif 


args); 


} 


O 仅 在 调试 模式 下 打出 日 志 。 
#ifdef _DEBUG), 





你 可 以 随意 使 用 一 些 其 他 条 件 (而 不 是 


适合 对 应 的 应 用 即 可 。 


更 好 的 选择 是 使 用 CocoaLumberjack 这 样 的 第 三 方 库 。 我 们 已 经 在 


这 里 使 用 的 


安全 的 底线 是 ， 如 果 设 备 被 解锁 ， 则 可 以 访问 所 有 数据 。 即 使 设备 被 锁定 ， 也 可 以 访问 大 


多 数 (也 可 能 是 所 有 ) 数据 。 








这 是 警察 和 小 偷 之 间 的 比赛 。 人 们 只 能 期 望 两 边 的 工具 能 更 高 级 和 更 智能 ， 但 和 争 权 
比赛 将 永远 不 会 结束 。 这 一 切 归根 结 底 就 是 谁 来 领导 和 谁 来 追赶 。 


9.4 数据 共享 

共享 数据 和 处 理 传人 数据 时 遵循 的 简单 基本 规则 是 : 不 要 信任 对 方 。 

当 接收 数据 时 ， 总 是 进行 验证 。 应 用 对 数据 的 唯一 假设 应 该 是 ， 它 可 能 是 无 效 且 错误 的 。 
为 了 提高 安全 性 ， 要 求 数据 进行 签名 。 


同样 ， 因 为 不 知道 哪个 应 用 会 处 理 数据 ， 所 以 永远 不 要 发 送 敏感 数据 。 如 果 你 确实 需要 共 
享 敏感 数据 ， 那 么 提供 令 牌 ， 然 后 要 求 其 他 应 用 从 你 的 应 用 (或 服务 器 ) 请 求 数据 。 


TI Ab 
9.5 ”安全 和 应 用 性 能 
额外 添加 的 加 密 或 安全 措施 会 计 入 总 内 存 的 消耗 之 中 ， 同 时 还 会 增加 处 理 时 间 。 你 没有 办 
法 在 所 有 维度 上 进行 优化 ， 只 能 做 一 些 权 衡 。 


有 时 ， 并 非 必 须 使 用 2048 位 的 RSA 密 钥 ，1024 位 的 DSA 密 钥 也 许 就 已 经 足够 了 。 其 他 
时 候 ，Rijndael 这 样 的 对 称 加 密 算法 就 足以 保护 数据 的 安全 了 。 ° 


从 钥匙 串 检索 初始 值 可 能 会 导致 加 载 时 间 延 长 。 你 在 使 用 时 应 该 小 心 谨慎 。 
证 书 锁定 有 甚 自己 的 成 本 ， 有 可 能 会 减 慢 所 有 的 网 络 操作 。 


创建 和 验证 数据 签名 需要 计算 内 容 哈 希 ， 这 意味 着 会 产生 额外 的 内 容 传递 。 根 据 内 容 的 大 
小 ， 这 可 能 需要 较 多 时 间 ， 更 不 用 说 计算 和 验证 数字 签名 所 需 的 额外 时 间 了 。 


所 有 这 些 步 又 会 快速 合 加 起 来 。 也 许 你 有 了 世界 上 最 保险 和 最 安全 的 应 用 ， 但 如 果 仅 加 载 
程序 就 需要 30 分 钟 ， 估 计 也 没有 人 想 使 用 它 。 对 于 这 一 点 ， 即 使 5 秒 钟 都 可 能 对 用 户 体 
验 产 生 负 面 影响 ， 黄 者 永远 失去 用 户 ， 尤 其 在 其 他 应 用 可 以 满足 同样 需求 的 情况 下 。 


9.6 清 


为 了 保卫 应 用 免 受 攻击 ， 你 需要 了 解 有 关 恶意 攻击 和 其 他 相关 领域 的 知识 ， 这 些 知识 是 无 
法 完全 列举 出 来 的 。 在 测试 10S 应 用 的 安全 性 时 ， 你 应 该 对 照 表 9-2 中 的 清单 。 


表 9-2: 安全 清单 


milly 


at 
a 























































































































静态 代码 分 析 

是 否 使 用 了 NSLog YES/NO 
如 果 是 ， 仅 在 调试 模式 中 使 用 NSLog YES/NO 
所 有 的 URL 都 是 HTTPS YES/NO 




















TES: 注意 ， 使 用 加 密 技 术 会 使 应 用 在 App Store 中 的 审批 生效 过 程 变 得 更 长 ， 因 为 可 能 会 对 你 使 用 的 API 
产生 导出 限制 。 




























































































































































































本 地 文件 的 路 径 不 是 硬 编码 的 YES/NO 
检查 最 新 版 本 和 补丁 程序 的 依赖 关系 YES/NO 
没有 使 用 私有 API YES/NO 
TOR rp ti A AL BA AA YES/NO 
资源 中 没有 嵌入 私 钥 或 隐私 YES/NO 
没有 运行 不 到 的 代码 或 无 用 代码 YES/NO 
权利 是 正确 的 (没有 丢失 ， 没 有 附加 ) YES/NO 
如 果 使 用 connection:willSendRequestForAuthenticationChallenge: 方法 , 没有 useCrede YES/NO 
ntial:forAuthenticationChallenge: 的 直接 分 支 (没有 任何 代码 ) 

应 用 使 用 了 IDFV YES/NO 
应 用 使 用 了 IDFA YES/NO 
为 应 用 签名 设置 正确 的 配置 文件 /证书 YES/NO 
检查 SQL 注入 YES/NO 
运行 时 分 析 一 一 日 志 

仅 对 文件 执行 日 志 记 录 YES/NO 
定期 删除 日 志文 件 YES/NO 
执行 日 志 循 环 YES/NO 
日 志 中 没有 隐私 或 敏感 信息 YES/NO 
打印 栈 跟踪 时 不 会 记录 敏感 信息 ” YES/NO 
运行 时 分 析 一 一 网 络 

只 使 用 HTTPS URL YES/NO 
服务 器 有 针对 CRIME 攻击 的 实现 YES/NO 
服务 器 和 客户 端 应 用 有 针对 BREACH 攻击 的 实现 YES/NO 
客户 端 使 用 证 书 锁定 YES/NO 
设置 正确 的 缓存 策略 YES/NO 
运行 时 分 析 一 一 认证 

六 用 使 用 第 三 方 身份 验证 YES/NO 
应 用 使 用 自 定 义 身 份 验证 YES/NO 
第 三 方 验证 SDK 按照 此 清单 的 其 余部 分 进行 了 严格 的 审核 YES/NO 
登录 UI 隐藏 了 密码 YES/NO 
密码 不 可 复制 YES/NO 
应 用 实现 密码 YES/NO 
密码 存储 在 钥匙 串 中 YES/NO 
可 以 通过 服务 器 的 配置 更 改 身份 验证 工作 流 YES/NO 
运行 时 分 析 一 一 本 地 存储 

应 用 使 用 本 地 存储 YES/NO 
加 密 所 有 的 敏感 信息 YES/NO 
周期 性 地 清理 存储 YES/NO 
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运行 时 分 析 一 一 数据 共享 

应 用 使 用 共享 密 钥 库 保 存 常见 设置 YES/NO 
验证 深层 链接 URL YES/NO 
验证 任何 传人 的 数据 YES/NO 
不 向 未 知 应 用 共享 任何 敏感 数据 YES/NO 
使 用 应 用 扩展 时 ,设置 正确 的 群 组 ID YES/NO 
此 清单 是 以 过 去 的 安全 性 知识 和 以 下 资源 为 来 源 而 编写 的 : 




















(1) 苹果 公司 的 iOS 安全 文档 (http://apple.co/116xVil ) 

(2) OWASP 移动 安全 项 目 一 一 安全 测试 指南 (https://www.owasp.org/index.php/Projects/ 
OWASP Mobile Security Project - Security Testing Guide) 

(3) OWASP iOS 应 用 安全 测试 备忘录 (https://www.owasp.org/index.php/IOS. Application 
Security Testing Cheat Sheet ) 

(4) OWASP iOS 开发 者 备忘录 (https//www.owasp.org/index.php/IOS, Developer. Cheat, Sheet) 

(5) Stack Overflow, “iOS 6 安全 分 析 工 具 ” (http://security.stackexchange.com/questions/23564/ 
security-analysis-tools-for-ios-6 ) 

(6) iPhone 应 用 的 渗透 测试 : 第 1 部 分 (http://www.securitylearn.net/2012/02/12/penetration- 
testing-of-iphone-applications-part-1/) 和 第 2 部 分 (http://www.securitylearn.net/2012/04/20/ 
penetration-testing-of-iphone-applications-part-2/) 


9.7 ”小结 

仅仅 陪读 单个 章节 是 无 法 完全 理解 安全 性 的 。 本 章 从 多 个 角度 出 发 ， 简 要 介绍 了 安全 方面 
的 一 些 关键 概括 点 。 我 们 探究 了 在 实施 安全 措施 时 的 一 些 做 法 ， 并 阐述 了 它们 是 如 何 影响 
整体 的 应 用 体验 的 。 

你 在 开发 时 应 遵循 本 章 末尾 的 请 单 ， 确 保 在 应 用 中 、 在 服务 器 上 以 及 在 两 者 之 间 能 够 解决 
常见 的 安全 漏洞 ， 或 者 至 少 能 够 实施 明确 有 效 的 措施 。 
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E90: 处 理 异常 或 其 他 情况 时 。 
10: 如 果 登 录 过 程 被 破坏 ， 应 该 能 够 改变 认证 流程 。 例 如 ， 触 发 双 因素 身份 验证 ,添加 CAPTCHA, 或 
在 极端 情况 下 从 原生 的 登录 UI 切换 到 Web 登录 。 























第 四 部 分 


代码 之 外 





见 好 就 收 ， 过 犹 不 及 。 
=R 





我 们 已 经 从 应 用 内 部 深入 研究 了 应 用 ， 现 在 是 时 候 跳出 来 了 。 


这 一 部 分 涵盖 了 应 用 测试 、 工 具 ， 以 及 在 自然 环境 下 监控 应 用 。 我 们 将 讨论 如 何 利 用 获得 
的 数据 及 应 用 产生 的 埋 点 数据 来 跟踪 性 能 并 完善 下 一 个 发 布 版 本 。 
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illl is A dh 





测试 一 项 功能 、 一 个 组 件 或 一 个 应 用 与 实现 它 同样 重要 。 

开发 团队 编写 代码 来 涵盖 各 种 场景 ， 而 质量 保证 团队 确保 代码 能 够 如 期 运行 。 质 量 保证 团 
队 和 开发 团队 往往 有 重合 之 处 ， 例 如 ， 在 创业 团队 或 小 公司 里 ， 开 发 和 质量 保证 工作 常常 
由 相同 的 人 或 团队 来 完成 。 

在 本 章 中 ， 我 们 将 研究 测试 用 例 的 基本 概念 、 类 型 、 支 持 类 型 的 框架 、 测 试 自动 化 ， 以 及 
持续 集成 。 
假设 团队 遵循 某 种 开发 方法 ， 甚 至 是 牛仔 式 编码 ， 那 么 你 确实 需要 编写 测试 用 例 来 规范 化 
应 用 测试 。 


10.1. 测试 类 型 
1 试 类 型 与 其 预期 的 明确 目的 有 关 。 例 如 ， 如 果 目 的 是 测试 一 个 方法 、 类 或 组 件 ， 那 么 它 
可 能 是 单元 测试 。 类 似 地 ， 如 果 其 目的 是 测试 应 用 从 安装 部 署 到 所 有 的 功能 ， 那 么 它 可 以 
归 类 为 验收 测试 或 端 到 端 测 试 。 
与 其 罗列 各 种 各 样 的 测试 类 型 ， 不 如 关注 以 下 类 型 。 它 们 在 面向 用 户 的 应 用 中 是 至 关 重要 
的 类 别 ， 尤 其 是 iOS 应 用 。 
。 单元 测试 

在 模拟 环境 中 测试 一 个 独立 方法 来 保证 其 有 效 性 。 
。 功能 测试 

在 真实 环境 中 测试 一 个 方法 来 确保 准确 性 。 





























eu 
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性 能 测试 
测试 一 个 方法 、 模 块 或 完整 应 用 的 性 能 。 


10.2 定义 
以 下 的 定义 会 在 我 们 讨论 测试 时 派 上 用 场 。 


测试 用 例 

需要 进行 测试 的 一 个 场景 。 它 包含 一 个 方法 、 功 能 或 程序 执行 所 需 的 条 件 ， 测 试 场景 需 
要 的 一 系列 输入 变量 ， 以 及 一 个 期 望 行为 ， 其 中 包括 系统 的 输出 或 改变 。 

测试 夹具 

表示 进行 一 个 或 多 个 测试 用 例 前 所 需 的 准备 和 清理 阶段 。 其 中 包括 对 象 创 建 、 依 赖 项 设 
置 、 数 据 库 设 置 ， 等 等 。 

测试 套件 

包含 一 系列 测试 夹具 的 测试 用 例 ， 可 以 租 套 其 他 的 测试 套件 。 它 用 于 聚合 应 当 一 起 执行 
的 测试 用 例 。 

测试 运行 器 

执行 测试 并 提供 结果 的 系统 。 在 本 书 中 ，Xcode 是 一 个 图 形 化 的 测试 运行 器 。 命 令 行 工 
有 具 同样 可 以 启动 一 个 测试 自动 化 常用 的 客户 端 。 

测试 报告 

测试 成 功 或 失败 的 内 容 摘要 ， 如 有 必要 ， 还 会 附 上 错误 信息 。 

测试 覆盖 率 

衡量 测试 套件 进行 的 测试 数量 ， 并 可 以 发 现 应 用 未 被 测试 的 部 分 。 如 图 10-1 所 示 ， 在 
代码 层面 测试 时 ， 测 试 覆盖 率 报告 将 总 结 有 多 少 行 代码 被 测试 所 秦 盖 。 一 份 详尽 的 报告 
还 可 能 标明 哪 部 分 代码 是 未 经 测试 的 。 















































v m HPMainTabBarViewController.m amm 
[M| -[HPMainTabBarViewController initWithCoder:] 
回 -[HPMainTabBarViewController insights] 
回 -[HPMainTabBarViewController viewDidAppear:] 
回 -[HPMainTabBarViewController loadAds] 
[I] -/HPMainTabBarViewController viewDidLoad] © 
_.41-[HPMainTabBarViewController viewDidLoad] block invoke — 
_.41-[HPMainTabBarViewController viewDidLoad] block invoke 2 
O -[HPMainTabBarViewController didReceiveMemoryWarning] 
IM] -[HPMainTabBarViewController viewDidLayoutSubviews] 











10-1: 测试 覆盖 率 报告 

测试 驱动 开发 

测试 驱动 开发 是 一 种 反复 迭代 且 开 发 周期 很 短 的 软件 开发 流程 。 其 过 程 包 含 编写 自动 化 
测试 用 例 、 编 写 通过 测试 的 最 小 代码 集 、 重 构 代 码 以 符合 准 入 标准 。 
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10.8 ”单元 测试 
单元 测试 检查 单独 的 方法 、 一 个 或 多 个 模块 的 集合 ， 还 包括 隔离 互相 关联 的 数据 以 验证 有 
效 性 。 为 实现 隔离 ， 依 赖 项 将 会 以 模拟 数据 的 形式 提供 满足 测试 场景 的 行为 。 


虽然 测试 所 有 的 方法 (包括 属性 的 获取 方法 和 设置 方法 ) 相当 枯燥 ， 但 这 都 是 值得 的 。 如 
果 一 个 方法 调用 失败 了 ， 你 将 会 得 知 是 哪个 方法 出 现 了 错误 以 及 具体 出 错 的 原因 。 


Xcode 内 置 支持 XCTest (http://apple.co/1PIWsUa) 单元 测试 框架 。 














10.3.1 设置 


单元 测试 需要 设 定 一 个 测试 目标 。 如 果 工 程 中 还 没有 设置 一 个 测试 目标 ， 那 么 你 需要 依照 
以 下 步骤 创建 目标 。 


(1) 打开 XCode 中 的 测试 导航 器 菜单 。 
(2) 点 击 十 号 按钮 。 

(3) 选择 New Test Target 选项 。 

(4) 如 图 10-2 所 示 ， 输 入 新 目标 的 内 容 。 











Choose options for your new target: 





Product Name: | HighPerformance Tests 
Organization Name: | Gaurav Vaish 
Organization Identifier: com.m10v 


Bundle Identifier: com.m10v.HighPerformance-Tests 











Language: Objective-C kd 
Project: ”加 HighPerformance 
Target to be Tested:  * HighPerformance i 
Cancel Previous 














10-2: 添加 测试 目标 





Xcode 本 应 该 在 几 处 地 方 对 工程 进行 配置 。 
首先 ， 确 认 工 程 中 已 创建 新 目标 ( 见 图 10-3), 
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PROJECT 
HighPerformance 


TARGETS 


Ee HighPerformanceTests 


(Œ) actionRead 
(E) shareRead 
(Œ) documentProvider 


(E) documentProviderFile... 











10-3. 确认 测试 目标 





其 次 ， 在 XCode 中 通过 Product 一 Scheme 一 Manage Schemes 一 Select project 一 Edit 打开 
工程 的 sheme， 你 应 当 能 发 现 ，Test > Test 处 的 一 个 条 目 与 之 前 创建 测试 目标 时 输入 的 
Product Name 一 致 。 如 图 10-4 所 示 ， 在 本 例 中 是 中 HighPerformanceTests。 








| @ HighPerformance ) 本 iPhone 6 (8.1) 





Build 
Inf Arguments 
ld ^ 2 targets id ia 
Run 
> P Debug Build Configuration ^ Debug 
Test A 
Ml v4 oru Debugger LLDB 
Pre-actions 
Test Tests Test Apr c a Test L 
Y Post-actions » (7 HighPerformanceTests None 4 None e 


Profile 
P T Release 


Analyze 
* Boos 


Archive 
d p Release 














Duplicate Scheme | Manage Schemes... © Shared 











10-4; product scheme 测试 设置 


10.3.2 ”编写 单元 测试 
创建 目标 后 ， 我 们 可 以 开始 编写 第 一 个 单元 测试 了 。 





E 
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XCTest 要 求 创建 XCTestCase 的 一 个 子 类 来 表示 一 个 测试 夹具 (并 非 如 其 命名 表示 软件 工 
作 中 的 测试 用 例 )。 在 该 子 类 中 编写 测试 用 例 。 包 含 所 有 测试 夹具 的 测试 套件 在 运行 时 以 
特定 顺序 调用 方法 。 














+[setUp] 

这 是 测试 夹具 的 设置 方法 ， 在 类 中 所 有 的 测试 用 例 执行 前 被 调用 。 测 试 夹 具 所 有 的 通用 
初始 化 工作 在 此 完成 。 注 意 ， 这 是 一 个 类 方法 。 

-[setUp] 

这 是 测试 用 例 的 设置 方法 。 在 每 个 测试 用 例 运 行 前 被 调用 。 在 此 方法 中 完成 每 个 测试 用 
例 的 初始 化 工作 。 它 是 一 个 实例 方法 。 














-[testXXX] 

测试 夹具 的 所 有 实例 方法 ， 名 称 以 test 开头 ， 并 且 不 含 任何 参数 ， 都 对 应 一 个 测试 用 例 。 
-[tearDown] 

这 是 测试 用 例 的 清理 方法 ， 在 每 个 测试 用 例 执 行 完 后 被 调用 。 它 是 一 个 实例 方法 。 
+[tearDown] 





这 是 测试 夹具 的 清理 方法 ， 在 类 中 所 有 测试 用 例 执行 完 后 被 调用 。 它 是 一 个 类 方法 。 























如 果 在 执行 测试 方法 时 没有 发 生 任何 错误 或 异常 ， 那 么 就 表示 测试 用 例 是 成 功 的 。 你 可 以 
通过 SDK 提供 的 断言 宏 检 查 更 复杂 的 场景 ， 比 如 测试 niL、 测 试 等 价 性 ， 等 等 。 这 些 宏 以 
XCTAssert<AssertionType>! 形式 命名 。 例 如 ，XCTAssertEqual 用 于 测试 两 个 值 或 对 象 是 否 等 价 。 























图 10-5 展示 了 整个 执行 生命 周期 的 可 视 化 过 程 。 注 意 ， 可 以 多 次 调用 -[setup] 和 


[tearDown] 实例 方法 。 








对 每 个 
测试 夹具 


对 每 个 测试 用 例 


-[setUp] 方 法 
方法 
方法 


方法 











图 10-5: 测试 执行 的 生命 周期 





ü 


E 1: iOS Developer Library, “Assertions Listed by Category" (http://apple.co/1N2sS6g). 
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前 文 的 例 4-9 中 介绍 了 HPAlbum 类 ， 现 在 我 们 为 此 类 写 一 个 单元 测试 。 


例 10-1 HPAlbum 单元 测试 


@implementation HPAlbumTest @ 
- (void)testInitializer { @ 
HPAlbum *album - [[HPAlbum alloc] init]; 
XCTAssert(album, Q"Album alloc-init failed"); © 


j 


- (void)testPropertyGetters ( @ 
HPAlbum *album - [[HPAlbum alloc] init]; 
album.name = @"Album-1"; 
NSDate *ctime = [NSDate date]; 
album.creationTime = ctime; 


HPPhoto *coverPhoto = [[HPPhoto alloc] init]; 
coverPhoto.album = album; 


album.coverPhoto = coverPhoto; 
NSArray *photos = @[coverPhoto]; 
album.photos = photos; @ 


XCTAssertEqualObjects(Q"Album-1", album.name); 
XCTAssertEqualObjects(ctime, album.creationTime); 
XCTAssertEqualObjects(coverPhoto, album.coverPhoto); 
XCTAssertEqualObjects(photos, album.photos); (9 

} 


@end 
Q 测试 夹具 通常 将 类 以 类 名 + Test 的 形式 命名 。 
四 测试 用 例 是 一 个 实例 方法 ， 其 名 前 级 是 test， 方 法 名 表示 测试 的 内 容 。 
© XcTAssert 方法 用 于 断言 对 象 不 是 nil, 
O 另 一 个 测试 用 例 ， 用 于 测试 属性 的 获取 方法 。 不 要 在 一 个 测试 用 例 中 测试 多 个 方法 。 
© 测试 状态 前 的 对 象 设 置 和 需要 提前 执行 的 代码 。 
Q 测试 对 象 等 价 性 的 断言 。 如 果断 言 失败 了 ， 则 测试 用 例 不 通过 。 这 说 明代 码 中 存 有 
bug， 需 要 修复 问题 。 


10.3.3 (C808 EX 

单元 测试 很 重要 ， 但 如 何在 结束 测试 后 得 知 代 码 中 经 过 测试 的 部 分 和 未 经 测试 的 部 分 ? 在 
自动 化 单元 测试 或 功能 测试 中 ， 我 们 使 用 代码 履 盖 率 来 表示 经 过 测试 的 代码 的 百分比 。 
代码 覆盖 率 文件 可 以 直接 经 过 苹果 的 LLVM 代码 生成 器 产生 ， 并 且 你 可 以 在 Xcode 中 修 
改选 项 。 有 两 种 方式 可 以 达到 此 效果 ， 甚 一 允许 Xcode 生成 可 视 化 报告 ， 另 一 种 生成 基于 
XML/HTML 的 报告 。 

1. 集成 覆盖 率 报告 

开发 人 员 可 以 运行 测试 用 例 并 通过 Xcode 查看 可 视 化 报告 。 
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要 想 获得 Xcode HJ UNG gs SER. ERR PA AUR RAGE. Au 10-6 所 示 ， 
通过 菜单 进入 Product ^ Scheme 一 Edit Scheme (38 <), Æ scheme editor 对 话 框 中 ， 选 择 
左 侧 的 Test 项 后 选中 Gather coverage data, “Ja si: Done 按钮 保存 设置 。 











> Pa Mun Info Arguments Diagnostics 
argets 
Run Build Configuration | Debug ia 
>> Debug B 
" y Test Code Coverage Gather coverage data 
Debug Debugger Debug executable 
Profile 
"Y Release 





Debug Process As * 


Analyze 
>a Debug 


Archive 
> P Release 


> C HighPerformanceTests None © None $ 




















图 10-6: 在 Xcode 中 启用 覆盖 率 数据 收集 


然后 运行 测试 用 例 。 根 据 设置 ， 在 运行 测试 用 例 的 过 程 中 ，Xcode 会 收集 覆盖 率 的 详细 数 
据 。 你 可 以 通过 以 下 步骤 查看 覆盖 率 报告 。 


(1) 打开 导航 栏 中 的 报告 导航 栏 ， 如 图 10-7 所 示 。 





B gRO AO S so(g 
ELTE 7e 


vw + HighPerformance Today, 2:03 AM 
六 Build Today, 2:03 AM 
P Debug 12/27/15, 9:17 PM 
4^ Build 12/27/15, 9:17 PM 
7 Test 8/3/15, 12:58 PM 











图 10-7: 报告 导航 栏 

(2) 选择 最 近 一 次 运行 的 测试 。 

(3) 打开 覆盖 率 标签 。 

你 应 该 可 以 看 到 类 似 图 10-8 所 示 的 覆盖 率 报告 。 

















测试 及 发 布 | 259 





Tests 





C) Show Test Bundles | (9) 


Name Coverage 
> m HPMeasurableView.m 


» m HPChapter08_ChildViewController.m 
v m HPAppDelegate.m —XX 
[I] -HPAppDelegate application:didFinishLaunchingWithOptions:] 
[I] -HPAppDelegate applicationwiliResignActive:] 
[I] -HPAppDelegate applicationDidEnterBackground:] 
[I] -HPAppDelegate applicationWillEnterForeground:] 
[I] -IHPAppDelegate applicationDidBecomeActive:] 
[II] -HPAppDelegate applicationWillTerminate:] 
[I] -HPAppDelegate applicationDidReceiveMemoryWarning:] 
[I] -IHPAppDelegate application:willFinishLaunchingWithOfftions:] O | 
[I] -HPAppDelegate application:shouldSaveApplicationState:] 
[I] -HPAppDelegate application:shouldRestoreApplicationState:] 
[I] -HPAppDelegate application:didDecodeRestorableStateWithCoder:] 
[I] -IHPAppDelegate application:willEncodeRestorableStateWithCoder:] 
[I] -HPAppDelegate application:pertormFetchWithCompletionHandler:] 
[II] -HPAppDelegate application:openURL:sourceApplication:annotation:] 
[I] -HPAppDelegate application:handleActionWithidentifier:forRemoteNo... 
[II] -HPAppDelegate application:didReceiveLocalNotification:] 
[I] -HPAppDelegate application:didReceiveRemoteNotification:] 
[I] -HPAppDelegate application:didReceiveRemoteNotification:fetchCo... 
[I] -HPAppDelegate application:didRegisterForRemoteNotificationsWith.... 











图 10-8: Xcode 集成 的 测试 覆盖 率 报告 


如 果 点 击 任意 方法 旁 小 的 右 箭 头 〈 见 图 10-8)， 那 么 将 会 跳 转 到 源 代码 处 。 跳 转 后 你 可 以 
看 到 测试 通过 的 确切 代码 行 数 以 及 未 被 测试 覆盖 的 代码 行 数 ， 如 图 10-9 所 示 。 




















131 -(void)applicationDidReceiveMemoryWarning: (UIApplication *)application 
132) { 0 


gger w:ü"[applicationDidReceiveMemoryWarning] called"); ° 
tation 1 3 Lines not covered 一 一 > 


logEvent:@"App_ MemWarn"]; 






rumentatio 





#pragma mark - Extra Delegate Lifecycle Events 

~(BOOL) application: (UIApplication *)application willFinishLaunchingWithOptions: (NSDictionary *)launchOptions 

ji 

[HPLogger w:G"[willFinishLaunchingWithOptions] called"]; + d 

ratur YES) Lines covered ——> 


) 


-(800L)application: (UIApplication *)application shouldSaveApplicationState: (NSCoder *)coder 





0 
uldSav 
forkey:@ 


名 [HPLogg te] called"]; 
D [coder 
149 return YES; 





[sho 
:1 














图 10-9: Xcode 一 一 源 代码 集成 的 覆盖 率 报告 


2. 外 置 测试 率 报告 

你 也 可 以 生成 XML 或 HTML 格式 的 报告 。 当 你 在 持续 集成 环境 或 非 开 发 人 员 的 机 器 上 运 
行 测 试 时 ， 或 当 你 想 要 保存 报告 以 备 未 来 使 用 时 ， 这 非常 有 用 。 

为 了 生成 包含 覆盖 率 数据 的 报告 ， 你 需要 局 用 以 下 标记 。 

* Generate Debug Symbols 

在 编译 生成 的 库 中 引入 调试 符号 。 


* Generate Test Coverage Files 


生成 包含 覆盖 率 数据 的 二 进 制 文件 。 




















* Instrument Program Flow 


在 测试 用 例 运 行 时 检测 应 用 。 


推荐 使 用 自 定义 的 构建 设置 来 区 别 测试 覆盖 率 和 常规 构建 ， 因 为 前 者 可 以 在 常规 测试 中 组 
慢 进 行 。 图 10-10 展示 了 你 可 以 在 哪里 找到 这 些 构建 设置 。 














General Capabilities 
Basic Combined Levels 十 
Y Apple LLVM 6.1 - Code Generation 


Setting 


Debug Information Level 
Enable Additional Vector Extensions 


Y Generate Test Coverage Files 


Build Settings 


Build Phases 


Q 


a HighPerformance 
Compiler default > 
Platform default 人 


Enforce Strict Aliasing Yes > 

Y Generate Debug Symbols Yes > 

r Release Yes 2 
Generate Position-Dependent Code No 


«Multiple values» 人 


Build Rules 





Debug Yes > 
Release No 人 

Inline Methods Hidden Yes 2 

Y Instrument Program Flow «Multiple values» ¢ 

Debug Yes > 
Release No? 

Kernel Development Mode No 

Link-Time Optimization No 人 











图 10-10; Xcode 开启 代码 覆盖 率 设置 

这 些 设 置 会 在 工程 的 衍生 数据 文件 夹 中 生成 .gcno 和 .gcda 文件 。.gcno 包含 重建 基本 代码 
块 和 块 对 应 源码 的 详细 信息 。.gcda 包含 代码 分 支 转换 的 计数 。 

下 一 步 是 使 用 这 些 文件 生成 可 以 导 为 XML 或 HTML 格式 的 报告 。 

以 下 的 两 个 工具 非常 有 用 。 


e lcov 
从 多 个 文件 收集 覆盖 率 数据 到 一 个 统一 的 INFOFILE 文件 。 
e genhtml 


用 lcov 工 具 生 成 的 INFOFILE 文件 来 生成 HTML 报告 。 


这 些 工 具 默 认 没 有 安装 在 Mac OS X 系统 上 ， 也 不 包含 在 Xcode 命令 行 工 具 中 。 使 用 
MacPorts (https://www.macports.org) 或 HomeBrew (http://brew.sh) 来 安装 Lcov 软件 包 。 
用 例 10-2 中 的 代码 在 Xcode 的 build phases 中 添加 一 个 New Run Script Phase， 从 而 生成 报告 。 


例 10-2 Xcode 代码 覆盖 率 报告 的 生成 过 程 


lcov --directory "S[OBJECT FILE DIR normalj/S(CURRENT  ARCH]" 
--capture 
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--output-file "S(PROJECT DIR) /S(PROJECT. NAME) . info" 


genhtml --output-directory "S(PROJECT, DIR)/S(PROJECT. NAME) - coverage" 
"${PROJECT_DIR}/${PROJECT_NAME}. info" 


构建 工程 后 ， 你 可 以 在 名 为 «our. project name»-coverage 的 文件 中 看 到 覆盖 率 报告 。 如 
果 打 开 主 文件 index.html， 你 可 以 看 到 类 似 图 10-11 所 示 的 报告 。 




















LCOV - code coverage report 


Current view: top level - Core Hit ^ Total Coverage 
Test: HighPerformance.info Lines: 69 178 sB 
Date: 2015-04-04 23:08:50 Functions: 15 495E% 


| Filename | LineCoverages 
HPCache.m L— ———4 76.0% 19/25 Am 
HPInstrumentation.m i | 63.69. 7/11 66.7% 

HPLocationManager.h | | | 009. 0/1 00% 0/2 
HPLocationManager.m DL O 00% 0/4 00% 0/14 


Generated by: LCOV version 1.11 




























10-11; HTML 覆盖 率 报告 


10.3.4 异步 操作 

假设 我 们 想 要 测试 HPSyncService 类 。 它 有 执行 异步 网 络 操作 的 方法 ， 也 许 不 能 立即 返 

服务 器 响应 。 我 们 需要 更 精确 的 技术 来 测试 这 样 的 方法 。 

XCTestCase 内 置 支持 测试 异步 方法 ， 因 此 ， 你 无 须 使 用 花哨 的 代码 来 支持 异步 操作 。 

测试 异步 方法 的 步骤 如 下 。 

(1) 使 用 expectationWithDescription: 方法 来 获取 XCTestExpectation 实例 。 它 也 在 所 谓 的 
手动 模式 中 设置 XCTestCase。 在 这 种 模式 中 ， 测 试 方法 的 完成 并 不 会 记 为 测试 用 例 通过 。 

(2) 使 用 waitForExpectationsWithTimeout:handler: 方法 来 等 待 操作 完成 。 如 果 测 试用 例 没 
有 完成 ， 那 么 将 会 调用 回调 处 理 的 闭 包 块 。 

(3) 使 用 XCTestExpectation 对 象 的 方法 fulfill 来 表示 操作 已 经 完成 ， 等 待 结束 。 这 就 是 
第 一 步 中 提 到 的 手动 模式 。 

例 10-3 提供 了 测试 异步 操作 的 具体 代码 。 


Iz] 
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例 10-3 ”测试 异步 操作 


@implementation HPSyncServiceTest 
-(void)testFetchType WithId Completion { 
HPSyncService *svc - [HPSyncService sharedInstance]; 


XCTestExpectation *expectation 
= [self expectationWithDescription:Q"Test Fetch Type"]; @ 


[svc fetchType:Q"user" withId:@"id1" 
completion:^(NSDictionary *) { @ 


//… 验 证 数据 ,使 用 断言 © 
[expectation fulfill]; @ 


1 


[self waitForExpectationsWithTimeout:1 handler: © 
[svc cancelAllPendingRequests]; © 
n 
} 
@end 


Q 获取 一 个 可 以 被 满足 的 期 望 对 象 。 我 们 现在 处 于 验证 测试 用 例 的 手动 模式 。 
e 执行 要 被 测试 的 方法 ， 进 行 合理 的 设置 和 参数 赋值 。 
© 如 前 所 述 ， 使 用 XCTAssertXXX 宏 来 验证 数据 。 

Q 一 旦 完成 ， 将 期 望 对 象 标记 为 已 满足 。 
O 等 待 期 望 对 象 被 置 为 满足 ， 本 例 等 待 1 秒 。 

O 如 果 期 望 对 象 没 有 被 满足 ， 则 进行 清理 操作 。 在 本 例 中 ， 取 消 任 何等 待 的 操作 。 


10.3.5 Xcode 6 福利 : 性 能 单元 测试 

你 可 以 在 单元 测试 中 进行 性 能 测试 。 

XCTestCase 类 提供 了 方法 measureBlock (http://apple.co/1SHL41A)， 后 者 可 以 用 来 测量 一 
个 代码 块 的 性 能 。 

例 10-4 展示 了 如 何 使 用 measureBlock 来 测试 用 例 的 性 能 。 

例 10-4 单元 测试 中 的 性 能 


-(void)testObjectForKey Performance { 
HPCache *cache = [HPCache sharedInstance]; 


















































[self measureBlock:“{ 
id obj = [cache objectForKey:@"key-does-not-exist"]; 
XCTAssertNil(obj); 
}]; 
} 


图 10-12 展示 了 例 10-4 中 测试 执行 时 的 输出 








o 





测试 及 发 布 | 263 









d)testObjectForKey Performance { 








49 HPCache *cache = [: sharedInstance]; 
50 
ios [self measureBlock:^| Time: 0.010 sec (2% STDEV) € 
52 id obj = [cache objectForKey:O"key-does-not-exist"]; 
53 XCTAssertNil (obj): 
54 Me 
55 } 
56 
57  Gend 
58 
[] C^ Il z sx £1 | NoSelection 





Test Suite 'HighPerformanceTests.xctest' started at 2015-08-03 19:57:27 «0000 

Test Suite 'HPCacheTest' started at 2015-08-03 19:57:27 «0000 

Test Case '-[HPCacheTest testObjectForKey Performance]' started. 
/Users/vgaurav/Desktop/all/projects/xcode/HighPerformance/HighPerformanceTests/Core/HPCacheTest.m:51: Test Case 
'-[HPCacheTest testObjectForKey Performance]' measured [Time, seconds] average: 0.010, relative standard 
deviation: 1.392*, values: [0.010410, 0.010426, 0.010104, 0.010067, 0.010492, 0.010159, 0.010327, 0.010434, 
0.010384, 0.010246], performanceMetricID:com.apple.XCTPerformanceMetric WallClockTime, baselineName: "", 
baselineAverage: , maxPercentRegression: 10.000*, maxPercentRelativeStandardDeviation: 10.000*, maxRegression: 
0.100, maxStandardDeviation: 0.100 

Test Case '-[HPCacheTest testObjectForKey Performance]' passed (0.501 seconds). 














10-12; 性 能 单元 测试 的 输出 结果 


Xcode 同时 给 出 运行 时 间 的 平均 值 及 标准 差 。 你 可 以 为 测试 值 偏差 设置 一 个 基线 。 
点 击 measureBlock 方法 所 在 行 的 对 号 进行 设置 ， 如 图 10-13 所 示 。 








yg? 
s 40 
Performance Result Ou - (void)testobj ectForKe 
ats 42 HPCache *cache = [ 
fill Result: No Baseline "a id obj = [cache ob 
D XCTAssertNil (obj); 
ne Average: 0.01s "i } 
8| Baseline: No Baseline "6 
ju : 47 
| Max STDDEV: 10.00% O48) -(void)testObiectForKe 
et 49 HPCache *cache = [ 
Set Baseline 50 
jos: [self measureBlock 
f 52 id obj = [cach 
53 XCTAssertNil(o 
54 We 
55 } 
56 
57  Gend 
M 8 
| IST | 
Value: 0.010 (1.02%) Test Suite 'HighPerformanceTe: 
— M Test Suite 'HPCacheTest' star 




















图 10-13: 为 性 能 单元 测试 设 定 基线 


一 且 设 置 好 基线 ， 输 出 结果 将 不 仅 显 示 平 均值 和 标准 差 ， 还 将 显示 低 于 基线 的 最 差 结果 ， 
如 图 10-14 所 示 。 














Performance Result 


8 Average: 0.01s 
Baseline: 0.01s 


4 Max STDDEV: 10.0096 
t 


Edit 


Result: 1.844% worse (+2%) 





Smo 


Value: 0.010 (2.33%) 























40 
ou 

2 

43 

us 

45 } 

46 

47 
O48 (void) test! 

49 HPCache 

50 

51 [self m 

52 i 

53 XCT 

54 Me 

} 
Ge 

[] C^ fl 
DUAILtU/AXHNLUI MELA 
HPChapterViewContr 
nrafilina: /Meare 





图 10-14; 根据 基准 线 测量 


10.3.6 ”模拟 依赖 


之 前 测试 的 类 HPAlbum 是 应 用 中 最 简单 的 类 之 一 。 


它 并 不 依赖 其 他 子 系统 (如 网 络 或 持久 


化 层 ) 。 一 般 来 说 ， 编 写 测试 代码 的 人 会 遇 到 很 多 问题 。 
如 果 我 们 想 要 测试 例 4-11 中 的 HPUserService， 尤 其 是 userWithId:completion: 方法 ， 那 
会 怎么 样 呢 ? 它 与 HPSyncService 类 交互 ， 后 者 中 的 fetchType:withId:completion 从 服务 


器 获取 数据 。 考 虑 以 下 的 问题 。 
(1) 应 用 是 否 真 的 应 该 发 起 网 络 请 求 ? 


(2) 如 果 想 要 测试 各 种 场景 ， 如 何 告知 服务 器 返回 




















何 种 数据 ? 











(3) 是 否 需 要 建立 其 他 服务 器 来 返回 虚拟 的 数据 ?如 果 是 ， 如 何 让 网 络 层 灵活 可 配 ， 根 据 使 








用 环境 〈 即 生产 或 测试 ) 与 各 种 服务 器 进行 通信 ? 
即便 网 络 层 是 灵活 可 配 的， 哪怕 概率 很 小 ， 我 们 i 


的 应 用 中 ? 





支 如 何 保证 这 些 配置 不 会 带 入 正式 发 布 





也 许 你 还 有 更 多 需要 考虑 的 问题 ， 这 就 是 为 什么 我 们 需要 一 个 系统 来 模拟 依赖 关系 。 带 有 


模拟 依赖 的 测试 用 例 的 工作 方式 如 下 。 


(1) 配置 依赖 项 以 便 依照 提前 定义 好 的 方式 运行 ， 返回 特定 的 值 ， 或 根据 特定 的 输入 改变 至 


特定 的 状态 。 
(2) 执行 测试 用 例 。 
(3) 重 置 依赖 项 使 一 切 正常 工作 。 








依赖 项 在 -[setUp] 方法 中 配置 ， 在 测试 夹具 的 -[tearDown] 方法 中 重 置 。 


词汇 





在 讨论 具体 框架 和 代码 前 ， 我 们 先 来 了 解 一 些 词汇 。 
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* dummy/double 

















用 于 描述 模拟 测试 对 象 的 通用 词汇 ，double 共有 四 种 类 型 。 

* stub 
在 测试 期 间 提供 封装 好 的 数据 以 便 被 调用 。 它 并 不 与 应 用 的 其 他 部 分 交互 或 改变 至 
其 他 状态 。 当 组 件 已 经 设计 好 用 于 依赖 注入 时 ，stub 非常 实用 。 在 测试 时 ， 配 置 为 
在 特定 方式 下 工作 的 依赖 项 可 以 被 注入 组 件 。 

* spy 
捕获 并 使 参数 和 状态 信息 可 用 。 它 记录 根据 参数 调用 的 方法 ， 并 帮助 验证 正确 的 方 
法 调用 。 测 试 时 ， 获 取 原 始 对 象 并 创建 一 个 spy 对 象 来 监控 方法 调用 。 最 后 ， 验 证 
行为 。 

* mock 
在 受 控制 的 情况 下 模拟 一 个 真实 对 象 的 行为 。mock 对 象 只 为 测试 用 例 需 要 交互 的 方 
法 进行 配置 。 

* fake 
除了 底层 实现 不 同 ， 它 与 原始 对 象 的 工作 方式 一 模 一 样 。 例如 ， 模 拟 数 据 库 在 内 存 
中 存储 数据 ， 而 且 ， 与 一 般 的 数据 库 引 擎 不 同 ， 它 可 以 进行 快速 的 搜索 。 

* BDD 


Dan North 发 明 的 行为 驱动 开发 ， 是 测试 驱动 开发 的 一 种 扩展 。 与 测试 驱动 开发 类 似 ， 


行 








为 驱动 开发 测试 特定 的 功能 ， 但 也 验证 底层 的 行为 。 


例如 ， 通 过 测试 一 系列 的 资格 证 书 来 检查 登录 功能 。 给 定 一 个 正确 的 资格 证 书 ， 函 数 应 


MZ 








当成 功 ， 否 则 不 通过 。 一 个 测试 驱动 开发 的 方法 可 以 帮助 你 测试 这 种 例子 ， 但 如 果 你 想 


要 验证 这 个 行为 ， 即 该 组 件 确实 调用 了 数据 库 或 网 页 服务 ， 那 就 需要 使 用 行为 驱动 开 
Ro dummy 对 象 可 以 模仿 或 模拟 底层 的 行为 并 用 于 验证 该 行为 ， 这 是 行为 驱动 开发 的 
关键 部 分 。 

。 mocking 框架 
人 允许 创建 dummy 的 框架 。 至 少 提 供 了 创建 mock 对 象 的 能 力 ， 但 通常 被 认为 也 可 以 创 
建 spy 对 象 。 

OCMock (http://ocmock.org) 是 一 个 非常 好 的 mocking 框架 ， 支 持 创 建 mock 和 spy HR. 



































此 处 不 对 原理 进行 深入 讨论 ， 我 们 来 看 看 使 用 框架 的 关键 因素 。 
。 创建 mock XL 
使 用 OCMCLassMock 宏 来 创建 一 个 类 的 mock 实例 。 
。 创建 spy HR 
使 用 OCMPartialMock 宏 来 创建 一 个 spy 或 一 个 对 象 的 部 分 mock, 
* stub 功能 
使 用 OCMStub 宏 对 函数 进行 stub 操作 ， 实 现 什 么 也 不 做 就 返回 或 返回 一 个 值 。 




















L 








iE 2: 


See Martin Fowler, “Mocks Aren't Stubs” .(http://martinfowler.com/articles/mocksArentStubs.html) 
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。 验证 操作 
使 用 OCMVerify 宏 来 验证 一 个 底层 子 系统 是 否 以 特定 方式 进行 交互 例如， 一 个 特定 方 
法 是 否 以 特殊 的 参数 被 调用 )。 


例 10-5 展示 了 一 个 使 用 OCMock 框架 的 测试 用 例 。 


例 10-5 使 用 OCMock 编写 一 个 高 级 测试 用 例 
#include «OCMock/OCMock. h» 
#include «OCMock/NSInvocation«OCMAdditions.h» @ 


@implementation HPUserServiceTest 


-(void)testUserWithId Completion { 
id syncService = OCMClassMock([HPSyncService class]); @ 
OCMStub([syncService sharedInstance]).andReturn(syncService); © 


NSString *userId = @"user-id"; 
NSString *fname - Q"fn-user-id", 
*lname = Q"ln-user-id", 

*gender = @"gender-x"; 
NSDate *dob = [NSDate date]; 


data = Qf 
@"id": userId, 
Q"fname": fname, 
@"Lname": lname, 
@"gender": gender, 
Q"dateOfBirth": dob 


}; O 


[OCMStub([ssvc fetchType:OCMOCK ANY 
withId:OCMOCK ANY 
completion:OCMOCK ANY 

]) andDo:^(NSInvocation *invocation) { @ 


id cb = [pinvocation getArgumentAtIndexAsObject:4]; 
void (^callback)(NSDictionary *) = cb; 
callback(data); Q 

H; 


HPUserService *svc = [HPUserService sharedInstance]; 
[svc userWithId:userId completion:^(HPUser *user) { €9 
XCTAssert(user); 
XCTAssertEqualObjects(userId, user.userId); 
XCTAssertEqualObjects(fname, user.firstName); © 
Ie 其 他 的 状态 验证 














OCMVerify([ssvc sharedInstance]); 
OCMVerify([ssvc fetchType:@"user" withId:userId completion:[OCMArg any]]); © 


@end 
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@ OCMock.h 是 引入 框架 的 主要 头 文件 。 使 用 NSInvocation+OCMAdditions.h 是 因为 我 们 
需要 实现 特定 的 功能 。 

Q 模拟 HPSyncService 类 的 一 个 对 象 。 

Q 对 类 方法 sharedInstance 进行 stub 操作 ， 以 返回 之 前 得 到 的 mock 对 象 。 

O 为 测试 用 例 输入 。 

Q 对 实例 方法 fetchType:withId:completion: 进行 stub 操作 来 满足 特殊 行为 。 

Q 就 测试 用 例 而 言 ， 基 于 输入 数据 执行 代码 ， 我 们 不 需要 进行 网 络 请 求 或 执行 数据 库 搜 
索 ， 或 缓存 查询 。 

O 所 有 的 配置 完成 后 即将 测试 方法 ， 此 时 调用 userWithId:completion:, 

O 运行 完成 后 验证 状态 。 

O 验证 方法 以 特定 参数 值 被 调用 


单元 测试 背 后 的 理念 是 将 要 测试 的 方法 视 为 黑 盒 。 测试 包括 为 其 提供 需要 的 输入 ， 对 比 实 
际 输出 和 期 望 输出 ， 而 不 必 了 解 国 数 的 实现 。 这 一 步 在 例 10-5 的 第 8 步 中 完成 了 ， 这 也 是 
测试 驱动 开发 背后 的 理念 


第 9 步 更 加 深入 方法 来 测试 和 验证 方法 是 否 与 依赖 项 以 特定 方式 交互 (比如 ， 以 特定 参数 
值 调 用 函数 )。 这 是 构成 行为 的 部 分 ， 也 是 行为 驱动 开发 的 关键 。 
10.3.7 ”其 他 框架 


OCMock 只 是 可 用 的 框架 之 一 ， 表 10-1 总 结 了 你 可 以 选择 的 其 他 流行 框架 
表 10-1: iOS 单 元 测试 框架 




















o 





























框架 类 型 名 称 保持 器 GitHub URL 
mock 对 象 “ OCMock Erik Doernenburg — https://github.com/erikdoe/ocmock 
OCMockito Jon Reid https://github.com/jonreid/OCMockito 
匹配 器 ° Expecta® Peter Jihoon Kim _https://github.com/specta/expecta 
OCHamcrest Jon Reid https://github.com/hamcrest/OCHamcrest 
TDD/BDD 框架 Specta Peter Jihoon Kim _https://github.com/specta/specta 
Kiwi Allen Ding https://github.com/kiwi-bdd/Kiw 
Cedar Pivotal Labs https://github.com/pivotal/cedar 
Calabash Xamarin http://calaba.sh 


10.4 功能 测试 
单元 测试 极 大 改善 了 对 单一 方法 的 测试 ， 但 因为 对 这 些 方法 的 测试 都 是 在 隔离 环境 中 进行 的 ， 











注 3: Source: Mattt Thompson, “Unit Testing” (http://nshipster.com/unit-testing/). 

注 4: 用 于 创建 模拟 对 象 。 

注 5: 用 于 创建 声明 式 匹配 规则 。 

注 6: 推荐 以 expect(album.name).to.equal(@"Album-1") 替换 XCTAssertEqualObjects(Q"Album-1", album.name), 











这 要 求 在 每 个 测试 用 例 执 行 前 进行 清理 工作 ， 所 以 它们 对 整体 应 用 的 测试 并 没有 太 大 帮助 。 


这 就 是 功能 测试 出 现 的 原因 。 顾 名 思 义 ， 功 能 测试 确保 应 用 的 功能 与 预期 一 致 。 这 
里 谈论 的 不 是 技术 操作 的 单位 ， 而 是 人 为 操作 的 单位 。 例 如 ， 我们 不 说 “测试 
authenticateWithCredentials: 方法 ”， 而 说 “测试 身份 验证 功能 "， 其 中 涉及 数据 输入 、 
网 络 操作 、UI 更 新 以 及 其 他 的 组 件 交 互 。 

功能 测试 更 多 的 是 在 UI 上 测试 ， 我 们 将 应 用 视 为 黑 盒 。 这 里 不 是 模拟 对 象 ， 而 是 应 用 的 
实际 操作 。 


Instruments 支持 通过 UI 自动 化 来 进行 功能 测试 ，UI 自动 化 是 自动 化 的 UI 测试 的 简称 。 


10.4.1 设置 


Instruments 提供 一 个 名 为 Automation 的 分 析 模 板 ， 该 模板 用 于 创建 新 的 功能 测试 或 导入 现 
有 测试 。 


通过 Xcode — Open Developer Tool 一 Instruments 路 径 ， 从 Xcode 中 启动 Instruments。 选 
F% Automation 并 点 击 Choose 按钮 ( 见 图 10-15)。 




















Choose a profiling template for: W gv-ios8-1 (8.3) ? @ HighPerformance.app 


Standard Custom Recent 


Bum 


Blank Activity Monitor Allocations [ Automation ] Cocoa Layout Core Animation 





guy à > 
= e 
I 
Core Data Counters Dispatch Energy File Activity GPU Driver 
Diagnostics 


Automation 
EN This template executes a script which simulates UI interaction for an iOS application launched from Instruments. 





í Open an Existing File... - 


— — i 
Cancel | 








图 10-15. Instruments 中 的 Automation 模板 





图 10-16 展示 了 UI Automation 的 Instruments 界面 。 你 可 以 参考 图 10-16 进行 如 下 配置 。 

(1) 在 设备 或 模拟 器 中 打开 应 用 。 如 前 所 述 ， 这 是 运行 真正 的 应 用 而 不 是 单元 测试 中 的 模拟 
或 独立 代码 。 

(2) 切换 到 Display Settings。 

(3) is 试 重 命 名 为 其 他 更 可 读 的 名 字 。 例 如 ， 如 果 想 要 测试 自 定 义 视图 的 组 合 ， 你 可 以 将 

其 命名 为 Test_CustomViews_Compos.te, 























测试 及 发 布 | 269 





ene 


@  ! | E gv-ioss-1 (8.3) ) È High Y 


Instruments. 





Instruments. 
00:00:00 




















| = | Executes iOS Automation scripts. 
Ir Automation i 


加 B 回 
VI Automation 
| 


) {E Script 


2 var target = UlATarget. localTarget (); 
3 








4 


eec 





© (si [ts 


Script Is stopped 


Status 


Scripts 


Test CustomViews Composite 


[Add ~) (Remove 


Script Options 
Run on Record 
Stop when Run Completes 
Logging 
Continuously Log Results 
Choose Location... 


© 


(Export Traced Results... ) 








10-16: Ul Automation 一 一 设置 








下 一 步 是 在 设备 中 启用 UI Automation。 出 于 安全 考虑 ，UI Automation 在 设备 中 默认 为 关 
闭 。 你 可 以 通过 Settings app 一 Developer 一 UI Automation 一 Enable UI Automation 来 启用 
UI Automation。 图 10-17 展示 了 如 何在 设置 中 启用 。 




















@ Game Center 


© Twitter 
e Facebook 


ee Flickr 


e Vimeo 


Developer 


€ Chrome 


pas ColorKeyboard 


a Finance 


ge Ganale Mans 





00003 = iem 4 100968994  eeooo Tile + gn 
Settings « Settings Developer 
UJ iBooks 
INSTRUMENTS 
fa Podcasts 
Logging 


Ul AUTOMATION 


10096 G+ 





Enable UI Automation 





NETWORK LINK CONDITIONER 


Status 


IAD DEVELOPER APP TESTING 


Fill Rate 
Ad Refresh Rate 
Highlight Clipped Banners 


Unlimited Ad Presentation 


Off 


These settings affect testing of developer-mode apps 


only. 





10-17: 启用 设备 上 的 Ul Automation 


完成 以 上 步骤 后 ， 我 们 就 可 以 开始 编写 功能 测试 了 。 
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10.4.2 编写 功能 测试 


写 功 能 测试 的 方式 有 两 种 。 第 一 种 方式 是 手写 所 有 的 代码 。 第 二 种 方式 是 使 用 记录 器 生 
成 代码 ， 然 后 修改 代码 ， 我 们 在 此 使 用 第 一 种 方式， 


如 图 10-16 的 第 4 步 所 示 ， 点 击 Record 按钮 开始 记录 。 这 会 在 目标 设备 或 模拟 器 中 启用 
应 用 。 

让 应 用 在 你 想 要 测试 的 场景 中 运行 。 一 旦 完成 ， 点 击 Stop 按钮 。 

尝试 为 以 下 场景 创建 一 个 测试 脚本 。 

(1) 启动 示例 应 用 。 

(2) 在 Chapters 界面 部 分 点 击 Threads。 

(3) 输 入 1000 表示 循环 的 次 数 。 

(4) 点 击 任意 地 方 收 起 键盘 。 
(5) 点 击 Compute Thread Creation Time 按钮 。 

(6) 验证 结果 ， 它 应 当 是 "Average Creation: <time> usec" 的 格式 。 
(7) 抽取 并 纪录 创建 时 间 。 


UI Automation 测试 用 例 在 Javascript 中 进行 开发 ， 所 以 你 需要 熟悉 
Javascript。 你 可 以 在 iOS 开发 者 库 的 “iOS UI 自动 化 测试 Javascript 参考 ” 
中 看 到 接口 参考 说 明 (http://apple.co/1KhVvXa)。 


| 



































自动 生成 的 代码 与 例 10-6 中 的 代码 类 似 。 
例 10-6 UI Automation 一 一 使 用 记录 器 的 默认 代码 


var target = UlATarget.localTarget(); @ 





target.frontMostApp().mainWindow() 
. tableViews()[0].tapWithOptions({tapOffset:{x:0.45, y:0.62))); © 
target. frontMostApp().mainWindow( ) 
.textFields()[0].tap(); 
target. frontMostApp().keyboard() 
.typeString("1000"); © 
target.tap({x:111.50, y:308.50}); @ 
target. frontMostApp().mainWindow( ) 
.buttons()["Compute Thread Creation Time"].tap(); @ 


Q 获取 目标 ， 目 标 可 以 是 设备 或 模拟 器 
@@ 点 击 对 应 的 单元 格 。 

Q9 进入 循环 计数 。 

Q 点 击 任意 地 方 收 起 键盘 。 

© 点 击 相应 的 按钮 。 


好 极 了 ， 现 在 我 们 有 了 正常 运行 的 代码 ， 可 以 在 此 基础 上 改进 。 让 我 们 按照 如 下 方式 更 新 
代码 。 
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。 不 要 使 用 水 平 垂直 坐标 轴 点 击 表格 视图 ， 我 们 希望 可 以 确切 点 击 特定 的 单元 格 。 在 此 ， 
我 们 点 击 第 7 个 单元 格 。 我 们 将 使 用 UIATableView 对 象 的 cells() 方法 来 获取 所 有 的 单 














元 格 ， 然 后 点 击 第 7 行 (索引 号 为 6)。 
。 我 们 需要 验证 结果 并 记录 创建 时 间 。 为 达到 此 目的 ， 我 们 将 使 月 
并 跟踪 成 功 或 失败 。 


在 此 不 深入 介绍 这 些 接口 ， 更 新 后 的 代码 如 例 10-7 所 示 。 
例 10-7 UI Automation 一 一 具有 确定 性 点 击 和 结果 的 更 新 后 代码 


var target = UIATarget.localTarget(); 











target.frontMostApp().mainWindow() 
.tableViews()[0].cells()[6].tapO; @ 
target.frontMostApp().mainWindow() 
.textFields()[0].tap(); 
target.frontMostApp().keyboard().typeString("1000"); 
target.frontMostApp().mainWindow() 
.buttons()["Compute Thread Creation Time"].tap(); 


var msg = target.frontMostApp().mainWindow() 
.staticTexts()[0].label(); @ 


var l = msg. length; 
if(msg.indexOf("Average Creation: ") != 0) {© 





H UIALogger 来 记录 消息 


UIALogger .logFail("Did not find average creation at the start"); @ 


) else if(msg.indexOf(" psec") != (l - 5)) { 
UIALogger.logFail("Did not find usec at the end"); @ 
} else { 
var t = msg.substring(18, l - 5); 


UIALogger.logMessage("Thread creation took "+ t + " usec"); © 


UIALogger.logPass("Hurray! Success."); © 
} 


Q 点 击 索 引 值 为 6 的 单元 格 。 





O 获取 第 一 个 静态 文本 的 LabeL。 在 复杂 的 UI 中 ， 你 可 以 使 用 accessibilityLabel 来 取 





RESI 
© 验证 label 的 值 。 
O 如 果 label 值 不 正确 ， 则 记录 一 条 不 通过 的 消息 。 
O 如 果 通 过 ， 记 录 计 算 时 间 以 及 其 他 内 容 。 
O- 标记 测试 用 例 为 通过 。 














关闭 应 用 。 运 行 测试 用 例 。 查 看 Edtior Log 部 分 的 界面 (ILE 10-18)。 这 部 分 会 报告 所 有 


执行 步骤 及 日 志 消 息 ， 包 括 最 终 失 败 或 通过 的 结果 。 
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m Automation (£3 Editor Log e- 
Index ^ Timestamp Log Messages Log Type 
0 12:33:38 PM PDT target.frontMostApp().mainWindow().tableViews([0].cells()[6].tap() Debug 
1 12:33:38 PM PDT target.frontMostApp().mainWindow/().textFields()[0].tap() Debug 
2 12:33:39 PM PDT target.frontMostApp().keyboard().typeString("1000") Debug 
3 12:33:40 PM PDT target.tap({x:"111.5", y:"308.5"}) Debug 
4 12:33:41 PM PDT target.frontMostApp().mainWindow().buttons()["Compute Thread Creation Time"].tap() ^ Debug 
5 12:33:42 PM PDT Message -> ‘Average Creation: 2580 usec' Default 
6 12:33:42 PM PDT Thread creation took 2580 usec Default 
7 12:33:42 PM PDT Success 











10-18; UI Automation—Editor Log 


10.4.3 ”工程 结构 


你 可 能 已 经 发 现 ， 功 能 测试 可 以 使 用 一 个 庞大 的 Javascript 文件 或 分 为 几 个 文件 ， 每 个 文 
件 表示 一 个 特定 的 场景 。 我 推荐 一 个 场景 使 用 一 个 文件 。 但 是 谨 记 ，Instruments 一 次 只 能 
执行 一 个 文件 。 这 意味 着 ,测试 多 个 场景 需要 多 次 启动 应 用 。 


管理 所 有 测试 用 例 的 一 个 理想 方法 如 下 。 


(1) 创建 一 个 名 为 tests 的 文件 夹 来 存储 所 有 的 测试 用 例 。 
(2) 在 该 文件 夹 中 创建 一 个 名 为 allTestsjjs 的 文件 。 
该 文件 没有 属于 自己 的 代码 ， 只 是 使 用 扫 mport 导入 其 他 文件 。 
(3) 为 场景 分 组 创建 子 文件 夹 。 
每 个 场景 一 个 文件 。 
(4) 从 Instruments 中 调用 allTests.js 文件 。 


Instruments 提供 一 个 命令 行 接口 来 执行 功能 性 测试 。 这 在 持续 集成 或 自动 化 构建 管线 中 非 
常 有 用 ， 我 们 将 在 10.7 节 中 对 此 进行 简单 讨论 。 一 个 典型 的 执行 命令 如 例 10-8 所 示 。 


命令 行 接口 



























































例 10-8 Instruments 


$ instruments 
-t '/Applications/Xcode.app/Contents/Applications/Instruments.app/Contents/ 
PlugIns/AutomationInstrument.xrplugin/Contents/Resources/ 
Automation.tracetemplate' @ 
-w '(device-uuid)' @ 
-e UIASCRIPT '/path/to/project/tests/allTests.js' © 
-e UIARESULTPATH '/path/to/projet/test-results/' @ 


@ Automation 模板 的 路 径 。 

@ 设备 的 UUID 或 模拟 器 标识 符 。 运 行 instruments -s 来 获取 模拟 器 列表 。 
© UI 自动 化 测试 Javacript 文件 的 路 径 。 

Q 测试 结果 所 在 文件 夹 。 
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10.5 ”隔离 依赖 


运行 单元 测试 时 ， 建 议 对 依赖 项 进行 分 离 和 模拟 。 这 样 可 以 让 测试 结果 免 受 依赖 项 变化 的 
干扰 。 

运行 功能 性 测试 时 ，0CMock 等 常用 的 模拟 框架 均 不 可 用 。 在 运行 功能 测试 或 性 能 测试 时 ， 
你 需要 让 这 些 子 系统 插件 化 ， 能 够 在 每 次 测试 前 重 置 状态 ， 隔 离 因 网 络 、Core Data 等 操作 
引起 的 变化 。 


举例 说 明 ， 早 前 在 例 4-11 中 使 用 的 HPSyncService 是 一 个 与 服务 器 同步 数据 的 中 心 。 我 们 
需要 让 它 可 配置 化 ， 以 便 shareInstance 方法 返回 一 个 根据 场景 返回 结果 的 对 象 。 
有 两 种 办 法 可 以 达到 此 目的 。 
。 创建 一 个 子 类 ， 该 子 类 返回 测试 场景 需要 的 数据 。 之 后 使 用 方法 替换 或 创建 方法 
setSharedInstance， 从 而 将 所 有 操作 都 转向 该 子 类 的 一 个 对 象 。 
这 种 方法 的 优点 在 于 所 有 操作 都 在 济 试 过 程 中 完成 ， 一 切 尽 在 你 的 黎 握 。 
。 创建 一 个 根据 特定 场景 返回 数据 的 服务 器 。 我们 称 之 为 场景 服务 器 。 在 运行 测试 用 例 前 ， 
根据 将 要 测试 的 场景 配置 服务 器 。 
这 种 方法 的 优点 在 于 只 需要 对 应 用 做 最 小 的 配置 改动 ， 改 变 连 接 的 域名 和 卫 地 址 。 
注意 ， 服 务 器 可 以 是 嵌入 测试 过 程 的 服务 器 。 对 于 类 似 的 戏 入 HTTP 服务 器 ， 你 可 以 使 
用 CocoaHttpServer (https:Wgithub.com/robbiehanson/CocoaHTTPServer) 。 
两 种 方法 都 需要 使 用 构建 目标 或 scheme 的 自 定义 二 进 制 包 。 前 一 种 方法 需要 大 量 自 定义 
的 代码 来 测试 不 同 场景 ， 并 且 需 要 自 定 义 scheme 在 构建 时 针对 不 同 场景 插入 额外 的 代码 
和 数据 。 在 后 一 种 方法 中 ， 服 务 器 的 IP 地 址 需要 合理 配置 ， 你 可 以 在 调试 模式 的 构建 中 使 
用 预 编 译 宏 。 
另外 ，UI Automation 运行 时 并 没有 API 与 场景 服务 器 交互 。 此 时 你 需要 更 高 级 的 框架 ， 
比如 Appium (http://appium.io) 或 Calabash (http://calaba.sh) 。 















































Calabash 使 用 Ruby 开发 。 如 果 想 要 使 用 它 ， 你 需要 学 习 Ruby 语言 。 运 行 
Calabash 服务 器 还 需要 一 个 自 定 义 的 工程 目标 。 

SE iti, Appium 支持 多 种 语言 (https://github.com/appium/sample-code/tree/master/ 
sample-code/examples) ， 也 需要 自 定义 目标 。 它 使 用 WebDriver 协议 与 应 用 通信 。 























例 10-9 展示 了 根据 构建 设置 配置 场景 服务 器 以 运行 测试 用 例 的 代码 。 
例 10-9 使 用 场景 服务 器 提供 场景 驱动 的 啊 应 数据 


//HPSyncService.m 


#define MACRO_STRING_(msg) #msg 
#define MACRO_STRING(msg) MACRO_STRING_(msg) @ 


#ifndef HP_CUSTOM_REMOTE_SERVER @ 
NSString *host = Q"https://my-real-server.com"; © 
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#else 
NSString *host = @MACRO_STRING(HP_CUSTOM_REMOTE_SERVER); @ 
#endif 














// 用 于 Appium 的 someTest.js 文 件 ,采用 Chai 断 言 库 和 Mocha 测 试 框 架 代码 风 格 G 
describe("login", function() { O 
before(function() ( @ 
// 配 置 WebDriver 
35 





after(function() ( @ 
// 关 闭 WebDriver 
IDE 


it("should succeed with valid credentials", function() ( @ 
http.get('scenario-server.com/setup?scenario id-valid login' 
+ '&client id-some-unique-id'); © 
driver .elementByName('username').text('testuser'); @ 
driver .elementByName('password').text('testpass'); 
driver .elementByName('Login').click(); 


driver .waitForELementByName('profileImage').should.be.ok; @ 
DE 
DE 


Q 用 于 拼接 字符 的 一 个 辅助 宏 。 

O 检查 是 否 已 经 定义 远程 服务 器 。 

© 如 果 未 定义 远程 服务 器 ， 则 使 用 默认 (生产 环境 )。 

O 如 果 已 定义 远程 服务 器 ， 则 使 用 自 定 义 服务 器 。 

© 使 用 Appium 和 JavaScript 进行 功能 测试 。 

Q 测试 套件 。 这 个 套件 测试 所 有 的 登录 场景 。 

O 套件 的 before 和 after 方法 在 测试 中 只 被 调用 一 次 ， 使 用 beforeEach 和 afterEach 在 
每 次 测试 用 例 前 配置 及 在 测试 用 例 后 重 置 。 

Q it 定义 一 个 测试 用 例 或 场景 。 

Q 配置 场景 服务 器 以 响应 特定 场景 ，http 表示 HTTP 模块 。 服 务 器 的 URL 仅 是 象征 性 
的 ， 你 应 当 能 理解 。 

O 设置 应 用 的 UI，elementByName 方法 根据 accessibilityIdentifier 搜索 对 应 的 UIView 
对 象 (http://apple.co/1At24WP)。 

QD 验证 profileImage 是 否 已 被 载 和 人 。 


例 10-9 仅仅 是 示范 性 的 代码 ， 不 过 它 可 以 帮助 你 加 快 编写 功能 性 的 测试 用 例 。 


10.6 测试 及 组 件 设 计 
测试 和 测试 框架 可 能 会 影响 组 件 的 设计 。 


例如 ， 埋 点 子 系统 或 许 应 该 设计 成 单 例 ， 并 抽象 出 儿 个 配置 步骤 。 但 在 测试 时 ， 你 可 能 希 
望 它 可 以 使 用 一 个 测试 应 用 ID ， 而 不 是 与 生产 环境 的 应 用 分 析 混为一谈 。 同 样 ， 对 单元 测 
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试 来 说 ， 你 可 能 需要 重 置 所 有 的 初始 化 过 程 ， 而 真实 的 应 用 一 次 就 可 以 完成 。 因 此 ， 组 件 
需要 设计 为 可 重 置 的 。 

对 同步 服务 等 其 他 组 件 来 说 ， 你 可 能 希望 配置 不 同 的 主机 地 址 。 对 每 个 发 往 服务 器 的 请 
求 ， 你 可 能 需要 附加 一 个 场景 DD， 以 便服 务 器 可 以 正确 响应 。( 场 景 ID 可 以 通过 使 用 自 定 
义 头 部 来 完成 添加 ， 比 如 ， 使 用 X-Test-Scenario-ID 或 修改 请 求 本 身 。) 


第 一 种 方法 是 使 用 模拟 数据 或 方法 替换 。 但 是 这 需要 深入 了 解 组 件 。 而 且 ， 随 着 场景 中 越 
来 越 多 的 方法 需要 模拟 数据 或 动态 禁 换 ， 这 种 方式 会 变 得 腔 肿 和 难以 管理 。 

第 二 种 方法 是 让 所 有 组 件 可 配置 。 推 荐 你 使 用 可 重 置 的 组 件 或 使 用 依赖 注入 的 生成 器 模式 。 
一 个 可 重 置 的 组 件 意味 着 ， 要 么 组 件 不 是 单 例 ， 因 此 可 以 被 多 次 创建 而 没有 副作用 ， 要 
么 它 还 是 单 例 ， 但 是 通过 sharedInstance 方法 访问 ， 并 且 提 供 setSharedInstance 及 
tearDown 在 每 次 测试 用 例 完成 后 重 置 共享 状态 。 该 方法 可 声明 在 一 个 私有 头 文件 中 ， 该 文 
件 对 使 用 此 类 和 SDK 的 开发 人 员 不 可 见 。 

如 果 组 件 有 一 些 依赖 项 ， 那 么 组 件 不 应 该 通过 创建 实例 或 使 用 单 例 来 直接 使 用 这 些 依赖 
项 ， 而 是 应 该 提供 一 个 自 定义 初始 化 方法 或 使 用 生成 器 模式 ， 方 便 依赖 注入 。 

例如 ， 不 推荐 编写 例 10-10 中 那样 的 代码 ， 你 应 当 使 用 例 10-11 中 那样 的 代码 。 


例 10-10 非 注 入 依赖 
-(instancetype) init { 
if(self = [super init]) { 
self.logger - [Logger sharedInstance]; 
self.instrumentation - [Flurry sharedInstance]; 


















































j 


return self; 


15] 10-11 依赖 注入 
-(instancetype) initWithLogger:(Logger *)logger 
instrumentation:(Flurry *)flurry { 
if(self = [super init]) { 
self.logger - logger; 
self.instrumentation - flurry; 


j 


return self; 


j 


如 果 有 对 其 他 系统 的 依赖 ， 那 么 你 可 能 需要 创建 包装 器 。 这 有 助 于 在 测试 中 完整 模拟 依赖 
项 。 它 也 能 帮助 你 创建 可 替代 的 依赖 。 例 如 ， 与 其 如 前 文 那 样 将 FLurry 作为 依赖 项 ， 不 如 
定义 一 个 Instrumentation 协议 及 合适 的 方法 。 在 实际 操作 中 ， 你 可 以 创建 一 个 与 Flurry 
相关 的 FlurryInstrumentation 实现 。 如 果 将 来 要 切换 到 MixPanel， 你 可 以 创建 另 一 个 
MixPanellnstrumentation 实现 。 



































Instrumentation 协议 定义 如 下 : 
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(protocol Instrumentation <NSObject> 
-(void)logEvent:(NSString *)eventName params:(NSDictionary *)params; 


Qend 


简 而 言 之 ， 当 想 让 代码 及 应 用 更 适合 测试 ， 让 自动 化 测试 用 例 胜 于 手工 测试 时 ， 你 需要 提 
前 考虑 组 件 设 计 。 应 用 及 组 件 的 设计 可 能 会 受到 计划 使 用 的 测试 框架 的 影响 ， 因 此 ， 你 需 
要 预先 谨慎 选择 及 规划 。 


10.7 ”持续 集成 与 自动 化 


持续 集成 可 以 保持 代码 清晰 并 确保 构建 及 时 更 新 。 

一 个 典型 的 开发 迭代 过 程 ( 见 图 10-19) 涉及 开发 人 员 编 写 代码 ， 然 后 代码 会 被 提交 到 一 
个 版 本 控制 系统 (如 Git 或 Mercurial)。 每 次 提交 会 触发 构建 管线 ， 接 下 来 是 所 有 的 测试 
(单元 、 功 能 、 性 能 、 集 成 等 )。 功 能 测试 也 许 会 在 模拟 器 或 各 种 设备 上 运行 。 例 如 ， 当 测 
试 一 个 消耗 大 量 内 存 和 CPU. 资源 的 游戏 时 ， 你 不 能 只 在 运行 iOS 8.x 的 iPhone 6 Plus Fil 
试 ， 还 要 在 运行 iOS 7.x 的 iPhone 4S 上 测试 。 所 有 测试 通过 后 将 会 产生 一 个 内 部 发 布 的 二 
进 制 包 ， 该 包 用 于 自动 化 不 能 完成 的 手动 测试 。 如 果 一 切 顺 利 ， 质 量 保证 团队 会 对 二 进 制 
包 进 行 签名 以 发 布 到 App Store 商店 。 






































B 10-19: 持续 集成 

除了 自动 化 单元 测试 和 集成 测试 ， 持 续集 成 在 构建 服务 器 上 运行 ， 进 行 额外 的 静态 和 动态 
的 代码 分 析 、 测 量 和 分 析 性 能 、 从 源码 创建 文档 ， 以 及 帮助 加 快手 工 质量 保证 的 过 程 。 
开源 、 商 用 以 及 云 主机 的 解决 方案 可 以 帮助 你 实现 应 用 的 持续 集成 。 在 可 选择 范围 内 ， 我 
推荐 Travis (https://travis-ci.com) 或 Jenkins (http://jenkins-ci.org) 。 
Travis 是 一 个 商用 解决 方案 ， 可 以 对 Github 上 的 开源 项 目 免 费 试 用 。 安 装 非常 方便 ， 只 需 
添加 Travis 配置 文件 ， 声 明 项 目的 文件 引用 、 场 景 以 及 iOS 版 本 。 之 后 将 Travis 构建 引擎 
指向 仓库 。 
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Travis 使 用 xctool (https://github.com/facebook/xctool) 来 构建 项 目 、 执 行 测试 ， 并 且 为 项 
目 依赖 集成 CocoaPods。 

使 用 Travis 的 缺点 是 ， 它 不 能 提供 设备 和 操作 系统 组 合 的 完整 矩阵 。 编 写本 书 时 ， 它 只 能 
支持 iOS 8.1 的 真实 设备 。 

另 一 方面 ，Jenkins 是 一 个 开源 工具 ， 一般 用 于 任务 执行 管线 管理 ， 主 要 关注 持续 构建 和 测 
试 软件 项 目 ， 监 测 cron, procmail 等 外 部 任务 的 运行 。 构 建 iOS 应 用 时 需要 Xcode 插件 。 
使 用 Jenkins 的 优点 是 ， 整 个 任务 管线 尽 在 你 的 掌握 ， 你 可 以 选择 使 用 instruments 命令 行 
工具 或 更 高 级 的 xctool， 也 可 以 接触 到 运行 Jenkins 的 物理 设备 并 运行 各 种 任务 。 

持续 集成 本 身 就 是 一 个 宽泛 的 主题 ， 需 要 单独 讨论 。 参 见 John Ferguson Smart 的 Jenkins: 
The Definitive Guide (http://shop.oreilly.com/product/0636920010326.do, O'Reilly), LA f fif 
更 多 有 关 Jenkins 的 信息 。 

你 也 应 该 研究 Xcode Server， 将 其 作为 持续 集成 的 方案 之 一 。” 

因为 xcodebuild 在 项 目 衍生 数据 文件 夹 中 生成 的 覆盖 率 和 其 他 中 间 文 件 信 
息 只 在 编译 期 间 可 用 ， 所 以 需要 额外 的 步骤 与 持续 集成 工具 结合 。 

就 Jenkins 而 言 ， 你 需要 以 下 步 又。 

(1) 在 Xcode 中 ， 添 加 一 个 build phase， 将 衍生 数据 路 径 复制 到 一 个 预先 声 
























































明 的 文件 中 。 
(2) 在 Jenkins 中 ， 添 加 一 个 构建 步 又， 以 便 使 用 该 文件 夹 生成 XML 格式 的 
覆盖 率 报告 。 


r3 x 
10.8 ”最 佳 实践 
正如 其 他 章节 ， 单 元 测试 也 有 一 些 可 遵循 的 最 佳 实践 。 和 毕竟， 它 的 代码 用 来 测试 别 的 代码 
( 别 好 奇 又 该 谁 来 测试 测试 用 例 )。 
编写 单元 测试 时 ， 你 应 当 遵 循 以 下 的 最 佳 实践 。 
。 测试 所 有 的 代码 ， 包 括 所 有 的 初始 化 方法 。 
。 测试 参数 值 的 所 有 组 合 。 
例如 ， 如 果 一 个 方法 接受 三 个 参数 ， 每 个 参数 可 以 取 两 种 值 (为 简单 起 见 ， 假 定 为 有 效 
和 无 效 ) ， 那 么 你 应 该 有 2 x 2 x 2=8 种 测试 用 例 。 因 此 ， 最 终 共 有 8 种 场景 。 
你 可 以 从 数据 库 中 心 获 取 数 据 ， 或 使 用 随机 数据 来 覆盖 所 有 的 场景 。 使 用 人 工 制 造 或 机 
器 生成 数据 结合 的 技术 被 称 为 模糊 测试 。 
。 不 要 测试 私有 方法 。 将 被 测 方 法 视 为 黑 盒 。 
。 建议 消除 任何 的 外 部 依赖 ， 这 保证 你 可 以 轻松 驾驭 各 种 场景 。 








iE 7: iOS Developer Library, ^ About Continuous Integration in Xcode” (http://apple.co/1 Kt4iXu). 
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。 在 每 个 测试 运行 前 设置 状态 ， 并 在 执行 后 清理 。 
确保 每 次 测试 用 例 点 结果 不 受 其 他 测试 影响 。 
。 每 个 测试 用 例 应 当 是 可 重复 的 ， 相 同 的 输入 产生 相同 的 结果 。 
。 每 个 测试 用 例 必 须 使 用 断言 来 验证 测试 的 代码 通过 与 否 。 
。 完整 的 运行 应 当 启 用 代码 覆盖 率 报告 。 这 能 提供 代码 已 被 测试 及 未 被 测试 的 概览 ， 以 及 
哪个 组 件 有 较 好 地 禾 盖 ， 哪 个 组 件 需要 关注 。 
除了 这 些 指 导 方 针 ， 别 忘 了 运用 编写 代码 的 其 他 最 佳 实践 。 单 元 测试 也 是 代码 。 
现在 对 功能 测试 也 采用 同样 的 最 佳 实践 。 唯 一 的 区 别 是 ， 与 在 类 中 测试 方法 不 同 ， 功 能 测 
试 需要 编写 对 业务 场景 ， 用 户 场景 的 测试 用 例 。 
此 外 ， 对 功能 测试 而 言 ， 你 还 应 当 测 试 各 种 设备 和 操作 系统 的 组 合 。 例 如 ， 在 iPhone 5S 
中 的 10S 7 系统 上 测试 、iPhone 4S 中 的 iPhone 8.3 系统 上 测试 、iPhone 6 中 的 iOS 8 系统 
上 测试 。 测 试 的 组 合 越 多 ， 功 能 测试 的 设备 覆盖 率 就 越 高 ， 这 将 确保 应 用 可 兼容 各 种 硬件 
和 操作 系统 相关 的 场景 。 


性 能 测试 

为 了 追求 测试 代码 功能 性 的 相关 因素 ， 人 们 往往 忘记 测试 与 质量 相关 的 因素 ， 尤 其 是 性 能 。 
令 人 感慨 的 是 ， 时 至 今日 对 应 用 性 能 测试 的 关注 少 之 又 少 。 大 公司 对 性 能 的 调研 相当 次 
入 ， 有 一 系列 工具 和 库 用 于 测试 服务 器 性 能 ， 但 是 缺失 客户 端的 性 能 测试 工具 。 如 果 对 
“iOS 应 用 性 能 测试 ”这 个 短语 进行 快速 的 网 页 搜索 ， 你 只 能 看 到 大 量 无 关 的 广告 。 


绝 大 多 数 公司 建立 企业 内 的 工具 来 测量 和 改进 性 能 。 个 人 能 拥有 的 最 好 工具 就 是 
Instruments。 可 以 使 用 它 对 内 存 、CPU 以 及 耗 电量 进行 分 析 ， 定 位 内 存 泄漏 ， 等 等 。 但 要 
想 在 单元 层面 测试 性 能 ， 你 还 是 需要 编写 自 定 义 的 代码 。 

XCTest 提供 了 neasureBlock 方法 进行 块 的 基本 性 能 测试 。 不 过 测试 用 例 只 会 报告 通过 或 
失败 以 及 覆盖 率 ， 并 不 会 解释 方法 运行 的 性 能 因素 。 

即使 在 运行 单元 测试 时 测量 性 能 ， 你 也 有 可 能 得 不 到 真实 的 数据 ， 这 取决 于 依赖 项 如 何 被 
模拟 。 

简 而 言 之 ， 为 了 测试 代码 的 性 能 ， 你 需要 对 希望 被 测试 的 内 容 编写 特定 的 代码 。 

为 了 计算 运行 速度 ， 你 可 以 使 用 一 个 简化 的 计时 器 。 例 10-12 提供 了 一 个 可 用 来 计算 耗 时 
的 计时 器 。 该 计时 器 支持 舱 套 以 发 现 调用 栈 中 的 瓶颈 。 

例 10-12 跟踪 运行 速度 的 计时 器 


@interface HPTimer @ 













































































*(HPTimer *)startWithName:(NSString *)name; 


@property (nonatomic, readonly, assign) uint64 t timeNanos; 
(property (nonatomic, readonly, copy) NSString *name; 
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-(uint64 t)stop; 
-(void)printTree; 


(gend 


(interface HPTimer () 


(property (nonatomic, strong) HPTimer *parent; 
(property (nonatomic, strong) NSMutableArray *children; 
(property (nonatomic, assign) uint64 t startTime; 
(property (nonatomic, assign) uint64 t stopTime; 
(property (nonatomic, assign) BOOL stopped; 

(property (nonatomic, copy) NSString *threadName; 


Qend 


(implementation HPTimer 


*(HPTimer *)startWithName:(NSString *)name ( @ 


j 


NSMutableDictionary *tls - [NSThread threadDictionary]; 
HPTimer *top = [tls objectForKey:Q"hp-timer-top"; © 


HPTimer *rv - [[HPTimer alloc] initWithParent:top name:name]; 
[tls setObject:rv forKey:@"hp-timer-top"]; 


rv.startTime - mach absolute time(); 
return rv; 


-(instancetype)initWithParent:(HPTimer *)parent 


j 


name:(NSString *)name { 
if(self = [super init]) { 
self.parent = parent; @ 
self.name - name; 
self.stopped - NO; 
self.children = [NSMutableArray array]; 
self.threadName - [NSThread currentThread].name; 
if(parent) { 
[parent.children addObject:self]; 
} 
} 


return self; 


-(uint64_t)stop { 


self.stopTime = mach_absolute_time(); 
self.stopped = YES; 
self.timeNanos = [HPUtils 
nanosUsingStart:self.startTime end:self.stopTime]; @ 


NSMutableDictionary *tls = [NSThread threadDictionary]; 
[tls setObject:self.parent forKey:@"hp-timer-top"]; © 


return self.timeNanos; 
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-(void)printTree { 
[self printTreeWithNode:self indent:@""]; 
} 


*(void)printTreeWithNode:(HPTimer *)node 
indent:(NSMutableString *)indent { @ 
if(node) ( 
DDLogDebug(@"%@[%@][%@] -> %lld", indent, self.threadName, 
self.name, self.timeNanos); 
NSArray *children = node.children; 
if(children.count > 0) { 
indent = [indent stringByAppendingString:@" "]; 
for(NSUInteger i = 0; i < children.count; i++) { 
[self printTreeWithNode: [children objectAtIndex:i] indent]; 


} 
@end 


// 使 用 率 
-(void)someMethodA { 
HPTimer *timer = [HPTimer startWithName:Q"method-A"]; @ 
[obj someMethodB]; 
[timer stop]; © 
[timer printTree]; @ 





} 


// 在 某 个 其 他 地 方 
-(void)someMethodB { 
HPTimer *timer = [HPTimer startWithName:Q"method-B"]; @ 
// 做 些 事情 
[timer stop]; @ 
// 或 者 ,printTree (9 





} 


@ HPTimer 类 的 公共 API, 

@ startwithName 创建 一 个 新 的 定时 器 ， 用 于 计时 。 

© 计时 器 上 下 文 是 线程 的 局 部 对 象 。 在 示例 的 实现 中 ， 一 旦 创建 计时 器 上 下 文 ， 计 时 器 可 
以 在 任意 线程 停止 。 这 种 实现 可 以 变 为 让 计时 器 对 限制 线程 调用 stop 方法 ， 只 在 创建 
计时 器 的 线程 中 调用 。 

O 初始 化 一 一 设置 稍 后 使 用 的 可 视 化 层级 。 

© 使 用 之 前 创建 好 的 辅助 方法 计算 以 纳 秒 为 单位 的 时 间 间 隔 。 

@ 从 线程 局 部 存储 中 获取 当前 线程 的 计时 器 。 

O 美化 计时 器 树 输出 。 

© 使 用 计时 器 需要 调用 startwithName， 为 计时 器 赋予 一 个 有 意义 的 名 字 。 

© 运行 后 调用 stop 方法 。 

@ 输出 运行 耗 时 ， 包 括 任意 仍 套 的 计时 器 。 
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O FRAN TSVAH, Git SS. 
@ 如 上 ,停止 计时 器 。 
Q9 你 可 以 选择 输出 蔡 套 的 调用 树 。 


10.9 小结 


测试 应 用 与 实现 它 一 样 重要 。 单 元 测试 帮助 你 在 最 适合 的 层面 测试 ， 而 功能 测试 在 真实 运 
行 环境 中 监测 应 用 。 启 用 代码 覆盖 率 是 一 个 推荐 步 又 ,这 可 以 帮助 你 持续 检查 代码 中 哪些 
步骤 还 未 被 测试 。 

持续 集成 是 整个 发 布 环 市 中 不 可 或 缺 的 部 分 。 自 动 化 测试 解放 了 工程 师 团队 ， 尤 其 是 与 持 
续集 成 过 程 相 结合 时 ， 这 避免 了 应 用 测试 中 任何 人 工 操作 导致 的 错误 。 

基于 测试 和 持续 集成 的 背景 ， 现 在 你 已 可 以 让 应 用 开发 、 构 建 和 发 布 过 程 形成 一 体 。 最 小 
化 人 工 干 预 可 以 减少 不 经 意 的 失误 ， 并 且 这 解放 了 质量 保障 人 员 ， 从 而 他 们 可 以 去 完成 其 
他 的 任务 。 
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第 11 章 


工具 





到 目前 为 止 ， 我 们 已 经 介绍 了 实现 一 个 高 效 、 高 性 能 应 用 的 绝 大 部 分 重要 因素 。 本 章 将 探 
讨 一 些 用 于 分 析 和 调试 各 种 问题 的 工具 。 

在 上 一 章 中 ， 我 们 学 习 了 可 以 使 用 代码 测试 应 用 的 有 效 性 和 独立 功能 部 分 的 性 能 。 但 是 ， 
在 分 析 特 定 任 务 时 还 需要 特定 的 工具 ， 这 些 任 务 包括 : 

。 识别 和 验证 无 障碍 标签 

。 从 资源 利用 率 的 角度 分 析 应 用 的 运行 时 执行 性 能 

。 分 析 网 络 和 Core Data 的 使 用 情况 

。 分 析 演 染 性 能 

。 通过 自动 化 代码 执行 用 户 交 互 过 程 

。 4T aati H s 

可 用 的 工具 有 很 多 ， 但 本 章 将 着 重 关 注 以 下 工具 : 


。 苹果 公司 的 Accessibility Inspector 
。 苹果 公司 的 Xcode Instruments 

。 Square 公司 的 PonyDebugger 

。 XK72 公司 的 Charles 


我 们 先 看 看 Accessibility Inspector。 











11.1 Accessibility Inspector 


为 了 获得 更 多 的 用 户 和 赞誉 ， 应 用 应 当 支持 无 障碍 功能 。 此 外 ， 当 地 法 律 可 能 还 会 强制 要 
求 应 用 支持 无 障碍 。 例 如 ， 美 国 的 508 无 障 得 法 案 可 能 要 求 整个 应 用 或 应 用 的 某 些 部 分 必 
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须 是 无 障碍 的 。 在 其 他 情况 下 ， 目 标 用 户 可 能 也 会 决定 要 求 ， 比 如 ， 旅 行 或 医疗 应 用 就 应 
当 支 持 无 障碍 。 

除了 其 他 属性 ， 每 一 个 UIView (或 其 子 类 ) 对 象 可 以 具有 accessibilityLabel 和 accessibilityHint 
属性 。 这 两 个 属性 控制 提供 给 残障 人 士 的 内 容 。 

accessibilityLabel 提供 展示 给 用 户 的 帮助 文本 。 例 如 ， 启 用 VoiceOver 可 以 大 声 读 出 文 
本 内 容 。 而 在 accessibilityLabel 不 足以 满足 需求 时 ，accessibilityHint 提供 关于 UI 元 
素 的 额外 信息 。 


为 了 分 析 应 用 是 否 设置 了 正确 的 值 ， 你 可 以 使 用 Accessibility Inspector, 


Accessibility Inspector 可 以 查看 应 用 中 每 一 元 素 的 无 障碍 信息 。 有 两 类 检查 器 可 用 : 一 类 
是 在 Mac OS X 系统 中 与 Xcode 集成 的 检查 器 ， 另 一 类 则 在 iOS 模拟 器 中 使 用 。 


车 想 从 Xcode 启动 检查 器 ， 导 航 至 Xcode 一 Developer Tools 一 Accessibility Inspector。 要 
想 在 模拟 器 中 启动 检查 器 ， 需 要 打开 设置 应 用 ， 通 过 其 中 的 General 一 Accessibility 打开 
Accessibility Inspector 开关 ( 见 图 11-1)。 注 意 ， 在 这 两 种 情况 下 ， 你 只 能 测试 模拟 器 。 
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- < General Accessibility 
File Edit View Find Navigate Editor Product 


About Xcode e ) Wå iPhone 5s (8.3) 
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Behaviors Accessibility Inspector 
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Beal <> | RB Hionp 
Open Developer Tool > W Instruments 
Services @ iOS Simulator 


Hide Xcode @ Accessibility Inspector Speech > 
Hide Others © FileMerge 
Show All B Application Loader 





























Quit Xcode More Developer Tools... Larger Text Off > 





Bold Text 


© 
Button Shapes ( ) 














Increase Contrast > 
Reduce Motion Off > 
On/Off Labels ( jo 

















& 11-1: 启动 Xcode (A) 和 iOS 模拟 器 (4G) 中 的 Accessibility Inspector 


11.1.1 Xcode Accessibility Inspector 


Xcode 中 的 Accessibility Inspector 提供 在 Mac OS X. 上 运行 的 应 用 的 无 障碍 信息 。 绝 大 部 分 
通用 元 素 可 以 在 iOS 模拟 器 上 正常 工作 ， 毕 竞 模拟 器 也 是 在 OS X 系统 上 原生 泻 染 的 。 
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图 11-2 展示 了 Xcode Accessibility Inspector 的 实 操 效果 。 


Permissions (Tap to Request) 标签 的 按钮 。 注 意 ， 




















加 








中 高 亮 的 元 素 是 带 有 























和 UIButton 的 titleLabel 属性 值 一 致 。UILabet 则 与 text 属性 值 一 致 。 





一 个 自 定义 的 UI 元 素 必 须 提 供 


它 自己 的 accessibilityLabel, 


在 检查 器 中 accessibilityritle! 属性 的 值 














( BOR | 


Accessibility Inspector (Locked) 


Y Hierarchy 
Y AXApplication 
Y AXWindow:AXStandardWindow 
AXButton 

Y Attributes 
isAccessibilityEnabled 
accessibilityParent 
isAccessibilityFocused 
accessibilityRole 


accessibilityHelp 
accessibilityValue 
accessibilityRoleDescription 
accessibilityWindow 
accessibilityFrame 

Y Actions 
accessibilityPerformPress 





accessibilityTopLevelUIElement__<AXWindow:AXStandardWindow 
accessibilityTitle Permissions (Tap to Request) 


No Selection 
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<AXWindow:AXStandardWindow 
NO 

AXButton 


<nil> 
<nil> 
button 
<AXWindow:AXStandardWindow 
x=815.00 y=255.00 w=166.00 hj 


b 
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iOS Simulator - iPhone 5s - iPhone 5... 
Carrier > 11:39 PM = 
< Chapters Location 
Permissions NR 
(Tap to Request) 
Track Location Changes 
Location Changes 0 
Distance from 
: 0 
Times Square 
Open Map Print Accuracy Levels 
2u 
a 6G «€ 29$ 
Chapters Second Debug Log Settings 








11-2: 使 用 Xcode Accessibility Inspector 检查 iOS 模拟 器 


11.1.2 











注 1: 








具 与 文档 存在 差异 ， 它 在 
(http://apple.co/IMnmG46) , 




















AppleKit 文档 中 写作 accessibilityLabel, 详情 参见 Mac 开发 者 库 


iOS Accessibility Inspector 
在 iOS 模拟 器 中 打开 Accessibility Inspector 后 ， 你 将 会 看 到 一 个 浮动 窗口 ( 见 图 11-3). 
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Carrier > EFM = 


< Chapters Location 


Permissions 
(Tap to Request) NR 


Track Location Changes 


L 


Location Changes 0 


Distance from 
Times Square 


Open Map Print Accuracy Levels 


Accessibility Inspector 


a G w d 


Chapters Second Debug Log Settings 











11-3: iOS 模拟 器 一 一 Accessibility Inspector; HEJA 


如 果 点 击 X 图 标 ， 然 后 选择 界面 上 任意 的 控件 ， 检 查 器 将 会 展示 该 控件 的 无 障碍 
信息 。 如 图 11-4 所 示 ， 标 记 为 1 区 域 的 按钮 被 点 击 后 ， 检 查 器 会 展示 按钮 边界 和 
accessibilityLabel 值 (2 区 域 中 的 Label)。 同 时 ， 它 还 会 展示 Traits 值 (http:/ 
apple.co/1HLcHI11)， 该 值 标识 无 障碍 控件 的 类 型 以 及 应 如 何 对 其 进行 操作 。 可 以 通过 




















accessibilityTraits 属性 (http://apple.co/1JrQJb6) 对 其 进行 配置 。 
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Carrier > 11:47 PM = 加 


X Chapters Location 


Permissions 
Jl w 
Track Location Changes 


Location Changes 0 


Distance from 
Times Square 


Open Map Print Accuracy Levels 


[x] Accessibility Inspector 


Permissions (Tap to 
Request) 

Button 

{{16, 72}, {166, 48}} 


Qnraan Dhanna Qnraan hanna 











Chapters Second Debug Log Settings 





11-4: iOS 模拟 器 一 一 Accessibility Inspector; 展开 状态 


Accessibility Inspector 可 以 帮助 你 对 应 用 进行 快速 的 无 障碍 分 析 。 











使 用 UI Automation 可 以 对 UI 中 每 一 个 控件 ?对 应 的 无 障碍 属性 进行 自动 化 


测试 。 




















注 2: 使 用 UIAELement 对 象 的 elements 属性 ， 尤 其 是 UIAWindow 对 象 表 示 当 前 活跃 应 上 


























见 iOS 开发 者 库 (http://apple.co/leQy25x)。 


的 主 窗口 。 详 情 参 


























11.2 Instruments 
HEEZE, Instruments 是 在 运行 时 诊断 、 调 试 、 分 析 应 用 的 事实 上 的 工具 。 在 上 一 章 探 讨 
功能 性 测试 时 ， 我 们 简单 介绍 了 这 个 工具 。 


本 节 将 对 Instruments 进行 深入 讨论 。 可 以 通过 Xcode 一 Open Developer Tool Instruments 
路 径 打 开 它 ， 如 图 11-5 所 示 。 














区 CC File Edi View Find Navigate Editor Product 
About Xcode e ) Wi iPhone 5s (8.3) Ru 
Preferences... 3, | | 
Behaviors 

| a | < > | È HiahP. 
Open Developer Tool > D Instruments 
Services > Wè iOS Simulator 
Hide Xcode 98H @ Accessibility Inspector 
Hide Others x3eH © FileMerge 
Show All 县 Application Loader 
Quit Xcode gQ More Developer Tools... 











11-5: 从 Xcode 中 启动 Instruments 


这 将 会 展现 一 个 包含 各 种 可 选 模板 的 菜单 ， 如 图 11-6 PAR. Xcode 6.3.2 中 集成 的 
Instruments 提供 了 约 20 种 不 同 的 分 析 模 板 ， 可 以 从 各 个 方面 来 诊断 应 用 和 设备 。 





Choose a profiling template for: (jg iPhone 5s (8.3 Simulator) » * HPerf Apps 


eos 


Standard Custom Recent 





v 


Core Data Counters Dispatch Energy File Activity GPU Driver 
Diagnostics 
| 
{ Leaks } Network OpenGL ES Sudden System Trace System Usage 
Analysis Termination 
Leaks 


Measures general memory usage, checks for leaked memory, and provides statistics on object allocations by class as 
d well as memory address histories for all active allocations and leaked blocks. 


Open an Existing File... Cancel | 




















11-6; Instruments 一 一 模板 选择 界面 
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选择 模板 后 按 住 AlVOption 键 ， 你 可 以 看 到 Choose 按钮 会 变 为 Profile 按钮 ， 
按 下 后 即 可 以 开始 分 析 。 





我 们 将 深入 讨论 一 些 更 有 趣 、 更 重要 的 模板 。 有 关 每 个 模板 的 细节 ， 参 见 iOS 开发 者 库 中 
HJ "Instruments 用 户 指南 ” (http;//apple.co/IRO7a6b) 。 


11.2.1 使 用 Instruments 


打开 模板 后 ， 你 会 发 现 Instruments 窗口 已 经 配置 好 相应 的 跟踪 器 。 你 可 能 希望 添加 更 多 分 
析 项 以 便 跟踪 。 每 个 分 析 项 被 正式 称 为 instrument, 


很 多 人 可 能 会 感到 困惑 。 工 具 本 身 名 为 mstruments， 而 监控 的 分 析 项 则 被 称 
为 instrument。 例 如 , “CPU” 是 一 个 instrument,“ 网 络 ” 也 是 。 因 此 ， 工 有 具 
IERRA Instruments 也 是 合情合理 。 

















用 Instruments 分 析 应 用 、 改 进 性 能 包括 以 下 步骤 。 


(GD 打开 一 个 模板 。 你 可 以 使 用 预先 定义 好 的 模板 ， 也 可 以 创建 一 个 空白 的 模板 。 
(2) W#INZ instrument， 这 一 步 是 可 选 的 。 

(G3) 分 析 应 用 ， 这 可 能 需要 启动 应 用 。 

(4) 收集 数据 。 

(5) 分 析 数 据 。 

(6) 如 有 必要 ， 更 新 应 用 。 

(7) 重复 上 述 步骤 直到 应 用 性 能 令 人 满意 。 


如 果 选 择 空白 的 模板 ， 那 么 你 将 会 看 到 一 个 空 无 一 物 的 Instruments 窗口 。 其 他 情况 则 会 根 
据 选择 的 模板 预先 选 好 对 应 的 instrument 项 。 


如 图 11-7 中 的 主 窗口 所 示 ， 点 击 窗口 中 的 Library 图 标 将 展开 列表 ， 你 可 以 选择 一 个 或 多 
个 instrument 进行 跟踪 。 11-8 展示 了 Library 列表 中 的 选择 界面 。 


使 用 Target 选项 选择 要 分 析 的 应 用 。 点 击 Record 按钮 将 会 开始 分 析 应 用 。 使 用 Pause/ 
Resume 按钮 可 以 暂停 或 恢复 记录 过 程 。 一 旦 完成 记录 ， 点 击 Stop 按钮 。 


主 窗口 右 下 方 的 区 域 称 为 检查 器 面板 ， 用 于 配置 记录 和 显示 设置 。 
窗口 左下 方 的 区 域 是 详情 面板 ， 用 于 展示 窗口 上 半 部 分 选中 的 instrument 项 的 相关 数据 。 
图 11-7 展示 了 Instruments 工具 的 主 窗口 ， 我 们 可 以 近 距 离 查 看 截屏 展示 的 各 种 选项 。 


(1) Library 图 标 (打开 了 instrument 列表 ， 如 图 11-8 所 示 )。 

(2) Target 选择 器 (可 以 选择 单独 的 应 用 或 设备 ， 你 可 以 只 选择 在 个 人 Mac OS X 设备 上 通 
过 开发 者 证 书 安装 的 应 用 )。 

(3) 记录 、 暂 停 、 恢 复 和 停止 按钮 。 


















































(4) 检查 器 面板 。 
(5) 检查 器 : 记录 设置 、 显 示 设 置 以 及 扩展 详情 。 
(6) instrument 选择 器 。 

(7) 按时 间 排 列 的 记录 图 。 

(8) 展示 已 选择 分 析 项 详情 的 详情 
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Examines a process’ heap for leaked memory; use with Allocations instrument to give memory address histories. 


Views recorded network activity logs. 
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11-7; Instruments 主 窗口 





e © Library 
|i] Library 
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e Reads/Writes - Records reads and 
writes of bytes to files. 








Library - Memory 


VM Tracker - Tracks the virtual memory. 
E space of a process over time, 
identifying regions by tag and... 


Shared Memory - Monitors the 
a opening and unlinking of shared | 
memory. 


= Leaks - Examines a process' heap for 
WR leaked memory; use with Allocations 
instrument to give memory address... 


Allocations - Analyzes the memory 
Nga life-cycle of process' allocated blocks; 
can record reference counting events. 
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Soak Leaks 


Examines a process' heap for leaked memory; 
use with Allocations instrument to give memory 
address histories. 























11.2.2. 活动 监视 器 

活动 监视 器 追踪 模板 监控 整体 的 设备 活动 (顾名思义 ， 包 括 CPU、 内 存 、 以 及 网 络 )。 
使 用 这 项 功能 可 以 发 现 应 用 过 度 使 用 任何 一 项 资源 的 情况 。 任 何 长 期 持续 的 过 量 消耗 系统 
资源 的 行为 尤其 邻 人 担忧 。 

如 果 发 现 一 项 资源 被 过 度 消耗 ， 那 么 必须 使 用 对 应 的 instrument 做 进一步 诊断 。 


该 模板 可 用 于 收集 一 系列 设备 层面 的 统计 数据 。 在 分 析 开 始 前 ， 在 检查 器 面板 中 选中 
Record Settings 项 ， 然 后 选择 ae FAM. 


图 11-9 展示 了 一 些 可 用 的 统计 项 ， 底 部 的 Select statistics to list 模块 展示 了 可 以 观察 的 所 
有 统计 项 ， 而 第 二 部 分 的 System statistics 模块 则 以 图 例 显示 选中 的 统计 项 。 
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图 11-9: 活动 监视 器 统计 项 选择 页 面 

































































B B ^ 
分 析 结 束 后 ， 你 应 该 可 以 看 到 类 似 图 11-10 的 结果 。 
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11-10: Instruments 一 一 活动 监视 器 


最 上 方 的 模块 展示 了 统计 项 的 用 时 。 图 
分 比 、 负 载 总 量 占 比 和 系统 负载 百分比 。 





11-10 显示 了 图 11-9 中 选中 的 线程 数 、 用 户 负 载 百 


因为 活动 监视 器 是 在 设备 层面 运行 的 ， 所 以 详情 面板 中 显示 的 饼 状 图 和 线 状 图 数据 都 是 快 


照 。 它 展示 了 四 个 图 表 。 


96 CPU 
消耗 CPU 资源 最 多 的 五 个 应 用 。 
CPU Time 








运行 时 占用 CPU 时 间 最 多 的 应 用 。 这 部 分 通常 没什么 用 ， 因 为 系统 应 用 总 是 位 居 榜 首 。 
e Real Memory Usage ( 饼 状 图 ) 

占用 内 存 最 多 的 五 个 应 用 。 你 不 会 希望 自己 的 应 用 长 期 出 现在 这 个 列表 中 。 
e Real Memory Usage ( 线 状 图 ) 





同样 展示 了 占用 内 存 最 多 的 五 个 应 用 ， 但 以 线 状 图 





11.2.3 内存 分 配 





展示 。 


内 存 分 配 追 踊 模 板 可 以 发 现 应 用 中 被 遗弃 的 内 存 。 被 遗弃 的 内 存 指 的 是 已 经 分 配 但 不 再 使 
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用 的 内 存 空 间 ， 这 部 分 空间 仍然 可 以 回收 。 

从 技术 角度 来 看 ， 被 遗弃 的 内 存 仍然 是 有 效 的 ， 因 为 它 被 应 用 的 某 个 部 分 引用 ， 只 不 过 实 

际 上 从 未 使 用 。 这 本 质 上 也 是 内 存 泄漏 ， 你 不 会 希望 在 发 布 的 应 用 中 出 现 这 种 情况 。 

分 配 了 内 存 空间 却 从 未 使 用 ， 这 在 内 存 消耗 不 高 时 似乎 不 算是 很 严重 的 问题 。 然 而 ， 如 果 

这 样 的 分 配 重复 出 现 ， 那 么 内 存 会 被 不 断 消 耗 ， 严 重 时 可 能 会 导致 内 存 不 足 ， 应 用 崩溃 。 

这 样 的 例子 包括 为 特征 对 象 分 配 内 存 但 从 未 使 用 的 未 完成 特征 。 这 只 是 一 个 简单 的 例子 ， 

在 现实 中 发 现 类 似 的 场景 其 实 远 比 这 复杂 得 多 。 

遵循 以 下 步骤 来 查找 被 遗弃 的 内 存 。 

(1) 选择 Allocations 模板 来 分 析 应 用 。 

(2) 确定 要 测试 的 初始 状态 。 

(3) 进行 操作 ， 让 应 用 从 初始 状态 到 另 一 个 状态 ， 然 后 返回 。 

例如 ， 这 些 步 骤 可 能 是 登录 、 操 作 应 用 ， 然 后 退出 登录 。 如 果 是 一 个 新 闻 应 用 ， 你 可 能 
会 选择 特定 的 类 别 、 阅 读 文章 ， 然 后 返回 类 别 的 页 面 。 

(4) f£ Display Settings 面板 中 点 击 Mark Generation 按钮 来 产生 一 个 堆 的 快照 ( 见 图 11-11) 。 
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11-11; 内存 分 配 一 一 Mark Generation 按钮 

(5) 多 次 重复 步骤 3 和 步骤 4。 

(0) 分析 快照 中 捕获 的 对 象 ， 以 便 定位 被 遗弃 的 内 存 。 
完成 以 上 步骤 后 ， 你 可 以 看 到 类 似 图 11-12 所 示 的 数据 。 


根据 图 11-12 中 最 上 方 的 图 表 ， 内 存 分 配 instrument 展示 了 内 存 使 用 率 随 时 间 的 推移 而 稳 
定 上 升 。 当 按 下 Mark Generation 按钮 时 ， 它 会 创建 一 个 版 本 的 快照 。 详 情 面板 展示 了 从 A 
到 EE 的 五 个 版 本 。 
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>< non-object > 
P Generation C 
PGeneration D 
» Generation E 
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Instruments16 
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) All Generations ~ € se [s 
Timestamp ^ Growth # Persistent | Generation Analysis ea 
00:14.576.242 18.40 MB 16,772 EAE 
00:18.524.994 9.64 MB 21 Ela Generation) 
48 Bytes Allocation Lifespan 
48 Bytes 1 3 
16 Bytes 1 5 
192 Bytes 3 ad 
- : 
00:21.523.041 9.63 MB 34 | Allocation Type 
00:24.960.253 9.63 MB 7 © All Heap & Anonymous VM 
00:27.892.157 9.69 MB 199 >All Heap Allocations 
All VM Regions 
Call Tree 
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在 示例 中 ，B 版 本 到 王 版 本 表明 每 个 快照 都 增加 了 约 9.5MB 的 内 存 。 展 开 B 版 本 的 快照， 
我 们 可 以 发 现 异常 的 <non-object> 项 占用 了 约 9996 的 增长 内 存 。 在 典型 的 内 存 分 析 方 法 
中 ， 这 可 能 很 难 调试 ， 因 为 它 的 类 型 是 未 知 的 ， 我 们 可 能 永远 也 找 不 到 。 

pa 


点 击 <non-object> 项 的 箭头 来 查看 详情 ( 见 图 11-13). 


ayy 






























































eoe Instruments16 
e il E gv-ios... ) ? HighPerformance.app ^ Run4of4 00:00:17 + |) 加 Em i 
rr 1 [OO 
» hl Allocations 
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了 Allocations ) Qi € ) All Gen Generation B: < non-object > © | © s E 
# Address Category Timestamp Live Sizev Responsible Li... Responsible Caller Description 
| foxi06004. D aloe 0.62 Me ]00:17.456.754) = ene 
1 Oxi4eBbc800  Malloc8.00KB 00:15.927.446 e UlKit -[UiView(AdditionalL ay... Type: Malloc 
2 0x14dd833a0  Malloc 512 Bytes 00:16.439.337 。 UIKit -[UlViewAnimationState... Pointer: 0x106604000 
3 0x14dd67370 Malloc 512 Bytes — 00:15.934.892 « Foundation _ 85-[NSISEngine choo... Retain Count: 1 
4 0xi4de44b00 Malloc 512 Bytes — 00:15.923.708 。 UIKit -[UlRuntimeConnection. Size: 10092544 


5 0x170264d80 
6 0x17026c400 
7 0x175465300 
8 0x17526cc40 
9 0x17026bf00 
10 0x17026cb40 
11 0x1754657c0 
12 0x10545ff80 





Malloc 64 Bytes 
Malloc 64 Bytes 
Malloc 64 Bytes 
Malloc 64 Bytes 
Malloc 64 Bytes 
Malloc 64 Bytes 
Malloc 64 Bytes 
Malloc 32 Bytes 





00:16.248.667  * 
00:17.883.099 。 
00:18.134.380 o 
00:18.253.460 。 
00:18.123.324 。 
00:16.500.228 e 
00:18.124.162 © 
00:17.082.607 e 


. libdispatch.dylib 
... libdispatch.dylib 
. libdispatch.dylib 
. libdispatch.dylib 
libdispatch.dylib 
. libdispatch.dylib 
. libdispatch.dylib 
. QuartzCore 


-dispatch continuation... | 
dispatch, continuation... Stack Trace 
- dispatch, continuation... o 
dispatch continuation... 
dispatch. continuatio: 
dispatch. continuation... 
. dispatch. continuation... 
mem alloc 





malloc. zone malloc 














malloc 
-[_ NSArrayM insertObject:atindex:] 
-[TrashCan init] 











-[UlControl touchesEnded:withEvent:] 
-[UlWindow . sendTouchesForEvent:] 
-[UlWindow sendEvent:] 
-[UlApplication sendEvent:] 


-UlApplicationHandleEventQueue 


CFRunLoopDoSources0 
CFRunLoopRun 





If 


-[HPChapter17Tools_02AllocationsViewCon... 
-[UlApplication sendAction:to:from:forEvent:] 
-[UlControl . sendActionsForEvents:withEvent: 


-UlApplicationHandleEventFromQueueEvent 


— CFRUNLOOP. IS CALLING. OUT. TO. A ... 
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在 所 有 的 <non-object> 内 存 分 配 中 ， 我 们 感 兴趣 的 是 导致 内 存 消耗 的 主要 部 分 。 在 图 
11-13 中 ， 数 据 按 空 间 大 小 降序 排列 。 第 一 项 表明 内 存 消耗 是 在 -[TrashCan init] 方法 中 
产生 的 ， 这 在 Responsible Caller 一 栏 中 体现 了 。 在 检查 器 面板 的 扩展 详情 界面 展示 了 调用 
期 间 完 整 的 栈 追 足 ， 这 展现 了 例 11-1 中 的 调用 树 。 

例 11-1 Hol 

malloc zone malloc 

malloc 

-[. NSArrayM insertObject:atIndex:] 

-[TrashCan init] 


-[HPChapteri7Tools 02AllocationsViewCon...] 
-[UIApplication sendAction:to:from:forEvent:] 


栈 追 踪 表 明 应 用 中 的 某 个 事件 调用 HPChapteri7Tools 02AllocationsViewController 中 的 
-[TrashCan init] 方法 。 


这 是 对 修复 应 用 问题 非常 有 帮助 的 信息 。 


11.2.4 Waitin 


VUE iti iB E Be n] DB Ra A TOL, FPN Ftd. EE, ERR 
的 应 用 不 会 发 生 这 样 的 问题 。 该 模板 还 提供 按 类 区 分 的 对 象 分 配 的 统计 ， 以 及 所 有 主动 分 
配 和 泄漏 块 的 内 存 地 址 记录 。 


内 存 泄 漏 模板 由 内 存 分 配 和 内 存 泄漏 instrument 组 成 。 我 们 已 经 在 上 一 节 中 研究 了 分 配 


instrument。 


内 存 泄漏 分 析 工 具 可 用 于 发 现 不 可 达 的 内 存 引 用 。Objective-C 类 分 配 的 内 存 以 类 名 显示 ， 
C struct 等 其 他 引用 对 象 显示 为 匿名 实体 及 其 分 配 的 大 小 。 


使 用 内 存 泄漏 模板 相当 简单 ， 点 击 后 就 会 马上 开始 收集 数据 。 
图 11-14 展示 了 应 用 在 一 次 运行 中 出 现 几 处 内 存 泄 漏 的 情况 。 








































































































eoe Instruments 

@ Il Bt gv-ioss-1 (8.3) )  HighPerformance.app Run 3 of 3 00:00:32 + E =] Cl oO 
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luu) =] I] 
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‘Leaked Object i. Address Size Responsible Library Responsible Frame SOIN T 
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160 Bytes | HighPerformance 
160 Bytes HighPerformance 





|-[HPChapter17Tools O3Leaks... 


-[HPChapter17Tools, O3Leal 


— 89-[NSProcesslnfo operatin... 










malloc. zone. calloc 


calloc 


> NSCFStrng 4 « multiple > 144 Bytes Foundation 
NSXCredentialStorage 1 0x170241780 48 Bytes libsystem c.dylib strdup class createinstance 
NSMutableDictionary 1 0x17487e5c0 64 Bytes CFNetwork XTubeManager::XTubeManage.... INSObject allocWithZone:] 
FABCertificatePinner 1 0x17401bfcO 16 Bytes HighPerformance ^ -[FABNetworkTaskUtility initWi... -[HPChapter17Tools. 03LeaksViewControlle... 
NSOperationQueue 1 0x170038440 32 Bytes CFNetwork -[NSURLSession initWithConf.. [E] -utAppiication sendAction:to:trom:forEvent:] 


NSURLSessionConfiguration 
OS dispatch queue 
NSXCredentialStorage 
NSXCredentialStorage 

— NSDictionaryM 

OS. dispatch. queue 


1 0x14e53ecf0 
1 0x1700f2d80 
1 0x1740d6730 
1 0x17422d5cO 
1 0x174245640 
1 Ox1700f2e00 


320 Bytes CFNetwork 
128 Bytes libdispatch.dylib 
112 Bytes Foundation 
32 Bytes CFNetwork 
48 Bytes CFNetwork 
128 Bytes libdispatch.dylib 


-[NSURLSessionConfiguration... 


_0s_object_alloc_realized 


__39-[NSProcessinfo operatin... 
85-[ NSURLSessionLocal... 
+[NSURLSession sessionWith... 


0S_object_alloc_realized 


[E] -uicontrot _sendActionsForEvents:withEvent:] 
-[UIControl touchesEnded:withEvent;] 
-UIWindow sendTouchesForEvent:] 
-[UIWindow sendEvent:] 

-[UlApplication sendEvent:] 





j EJCICIOIICIICI 


NSXCredentialStorage 1 0x170251910 48 Bytes CFNetwork XURLCache::createNSXURLC... -UlApplicationHandleEventFromQueueEvent 
NSXCredentialStorage 1 0x17003c580 32 Bytes CFNetwork XTubeManager::withTubeMana... -UlApplicationHandleEventQueue 

— NSOperationQueuelnternal 1 Ox14e53ee30 384 Bytes Foundation -[NSOperationQueue init] CFRUNLOOP IS CALLING OUT. TO A ... 
NSXCredentialStorage 1 0x17003d740 32 Bytes libsystem c.dylib — strdup . CFRunlLoopDoSourcesD 
NSMutableDictionary 1 0x17487aa00 64 Bytes Foundation .89-[NSProcesslnfo operatin... CFRunicopRun 

NSXCredentialStorage 1 0x17422d5a0 32 Bytes CFNetwork __85-[_NSURLSessionLocal... EJ crruntooprunspecite 
FABNetworkTaskUtility 1 0x174225360 32 Bytes HighPerformance ^ -[FABDownloadAndSaveSettin... 

NSXCredentialStorage 1 0x17410cf90 144 Bytes. CFNetwork +[NSURLSession sessionWith... E53 cseventrunwodal 

NSXCredentialStorage 1 0x170037000 32 Bytes CFNetwork XTubeManager::withTubeMana... n UlApplicationMain 

— NSCFString 1 0x174244200 48 Bytes Foundation — 89-[NSProcessinfo operatin... main 

__NSURLSessionLocal 1 0x17014a500 176 Bytes CFNetwork +INSURLSession sessionWith... ^ | [E] start 
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图 11-14: AFW 3 
内 存 泄漏 分 析 工 具 在 顶部 的 展示 栏 中 显示 了 一 些 红 色 的 条 ， 那 就 是 发 生 内 存 泄漏 的 时 刻 。 
条 越 高 ， 内 存 泄漏 就 越 大 。 详 情 面板 中 列 出 了 哪些 对 象 发 生 泄漏 ， 以 及 泄漏 的 数量 和 沦 
漏 内 存 的 大 小 。 检 查 器 面板 中 的 扩展 详情 界面 展示 了 当时 的 栈 追 踪 ， 这 有 助 于 分 析 泄 漏 

















在 真实 的 应 用 中 ， 捕 获 内 存 泄漏 往往 是 非常 枯燥 的 过 程 ， 有 时 还 需要 几 个 小 
时 的 测试 。 

作为 练习 ， 你 可 以 检查 代码 每 次 更 新 后 
辑 层 面 ， 对 各 部 分 进行 内 存 汇 漏 分 析 。 


是 否 产 生 了 内 存 泄 漏 。 深 入 应 用 的 逻 


11.2.5 ”网 络 


网 络 追 踪 模 板 帮 助 你 分 析 应 用 产生 的 网 络 带 宽 连 接 。 一 旦 发 现 过度 使 用 网 络 、 多 域名 连 
接 、 多 次 下 载 相同 内 容 ， 以 及 使 用 不 安全 的 连接 等 情况 ， 则 必须 着 手 处 理 。 
该 模板 由 连接 instrument 组 成 ， 图 11-15 展示 了 使 用 网 络 追 踪 模 版 的 结果 。 详 情 面板 提供 


了 调试 网 络 使 用 情况 时 需要 的 内 容 ， 其 中 包括 远程 服务 器 地 址 i Pee 
则 会 显示 域名 ) 、 传 输 数据 量 、 请 求 往返 的 平均 时 长 和 最 短 时 间 ， 等 等 。 
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Q Il Bi gv-ios8-1 (8.3) ) @ HighPerformance.app | Run 4 of 4 00:01:05 +E [ss] 
| Instruments TTT dokfot TT dobdo! TT To | * dodo 11 lodo t TTE Molde loieto t TTT 
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L PENS - = = aos 一 一 | 
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HighPerformance (1976) 254.50 KB tcp4 p3sIh005.shr.phx3.secureserver.... tcp4 192.168.0.28:50051 192 495KB 6 OBytes OBytes 0 
HighPerformance (1976) 242.69 KB tcp4 a23-203-243-215.deploy.static.a... tcp4 192.168.0.28:50033 179 600KB 15 OByles OBytes 0 
HighPerformance (1976) 212.40 KB tcp4 p3sih005.shr.phx3.secureserver.... tcp4 192,168.0.28:50050 163 5.84KB 7 OBytes OBytes 0 
了 Other 195.99 KB udp4 *:* udp4 *:5353 1720 738KB 86 
HighPerformance (1976) 172.14 KB. tcp4 a23-203-243-215.deploy.static.a... tcp4 192.168.0.28:50035 126 350KB 9 792 Bytes OBytes 0 
HighPerformance (1976) 142.50 KB tcp4 93.184.215.200:80 tcp4 192.168.0.28:50030 101 362 Bytes 1 OBytes OBytes 0 
7 “Other 125.33 KB udp6 ::.* udp6 ::.5353 1303 7.38KB 86 
HighPerformance (1976) 86.05 KB — tcp4 a23-203-243-215.deploystatic.a... tcp4 192.168.0.28:50043 62 359 Bytes 1 OBytes OBytes 0 
HighPerformance (1976) 85.20 KB tcp4 a23-203-243-215.deploy.static.a... tcp4 192.168.0.28:50032 64 102KB 3 93Bytes OBytes 3 
HighPerformance (1976) 79.97 KB tcp4 a23-203-243-215.deploystatic.a... tcp4 192.168.0.28:50042 58 356 Bytes 1 OBytes Bytes 0 
^! HighPertormance (1976) 60.07 KB tcp4 a23-203-235-92.deploy.static.ak... tcp4 192.168.0.28:50001 47 173KB 5 OBytes UBytes 0 
HighPerformance (1976) 58.30 KB tcp4 a23-203-232-239.deploystatic.a... tcp4 192.168.0.28:49997 43 663 Bytes 2 OBytes OBytes 0 
^. HighPerformance (1976) 54.87 KB — tcp4 a29-203-232-239.deploy.static.a... tcp4 192.168.0.28:49996 42 116KB 2 143 Bytes OBytes 0 
HighPerformance (1976) 51.31 KB tcp4 a23-203-232-230.deploystatic.a... tcp4 192.168.0.28:49908 38 595 Bytes 1 OBytes OBytes 0 
HighPerformance (1976) 50.53 KB  tcp4 a23-203-235-92.deploy.static.ak... tcp4 192.168.0.28:50002 38 171KB 5 OBytes OBytes 0 
HighPerformance (1976) 42.25 KB tcp4 93.184.215.200:80 tcp4 192.168.0.28:50031 31 342 Bytes 1 OBytes OBytes 0 
HighPerformance (1976) 40.29 KB — tcp4 a-0003.a-msedge.net:80 top4 192.168.0.28:49995. 33 310 Bytes 1 OByles OBytes 0 
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图 11-15: 网 络 一 一 连接 


11.2.6 时间 分 析 器 

时 间 分 析 器 追踪 模板 以 固定 频率 收集 栈 追 踪 数 据 。 它 由 同名 的 instrument 组 成 。 

在 你 与 应 用 交互 时 ， 它 会 记录 方法 位 于 栈 顶 的 次 数 。 你 可 以 通过 这 个 数字 来 识别 哪 此 方法 
被 调用 多 次 或 者 占用 了 大 量 运行 时 间 。 如 果 某 个 方法 在 1000 个 样本 中 位 于 栈 顶 100 次 ， 
那么 可 以 假定 它 占用 了 约 10% 的 总 运行 时 间 。 

如 果 一 个 方法 在 应 用 运行 的 大 部 分 时 间 里 都 在 执行 ， 这 可 能 会 令 人 担忧 。 其 原因 可 能 涉及 
用 户 频 繁 操作 或 应 用 已 经 超 负 蓓 工作 。 根 据 应 用 的 类 型 ， 这 可 能 是 一 个 需要 关注 的 领域 。 
例如 ， 游 戏 应 用 可 能 需要 用 户 频 繁 操 作 ， 但 工具 类 应 用 则 可 能 不 需要 。 类 似 地 ， 视 频 或 持 
续 的 动画 将 会 消耗 大 量 资 源 ， 但 邮件 类 应 用 则 不 会 。 


图 11-16 展示 了 使 用 时 间 分 析 器 追踪 模板 捕获 的 数据 。 
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在 图 11-16 中 ， 栈 追踪 展示 ， 如 果 应 用 运行 缓慢 ，-[HPMeasurabLeview layoutSubviews] 很 
可 能 是 罪魁 祸首 。 


11.3 Xcode 视图 调试 器 


随 着 越 来 越 多 的 功能 加 入 应 用 ，UI 变 得 越 来 越 复杂 ， 不 仅 设计 和 实现 变 得 更 加 复杂 ， 演 染 
性 能 也 会 受到 影响 。 随 着 泻 染 的 视图 层级 舱 套 得 更 复杂 (或 层级 中 视图 数量 的 增加 )， 泻 
染 、 更 新 或 运行 动画 (包括 展示 UIPickerView 或 在 UIScrollView 或 UITableView rjf z) ) 
等 行为 的 用 时 都 会 增加 。 

Xcode 的 视图 调试 器 可 以 帮助 你 在 应 用 运行 期 间 查 看 视图 层级 。 


在 Debug 区 域 工具 栏 中 点 击 Debug View Hierachy 按钮 可 以 激活 视图 调试 器 。 点 击 后， 主 
线程 将 会 暂停 ，Xcode 中 的 主要 视图 区 域 就 会 展示 当前 视图 层级 的 快照 。 你 可 以 拖 动 快照 
以 3D 视角 查看 ， 如 图 11-17 所 示 。 
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图 11-17: Xcode 的 视图 调试 器 一 一 视图 层级 














如 果 在 视图 层级 中 点 击 视图 元 素 并 打开 Utilities 区 ， 那 么 你 应 该 可 以 查看 更 多 关于 视图 的 
细节 (WLI 11-18), 
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11.4 PonyDebugger 
使 用 Xcode 视图 调试 器 时 ， 主 线程 将 会 暂停 。 因 此 ， 你 需要 暂停 运行 中 的 动画 才能 跟踪 视 
图 层级 。 


PonyDebugger (https://github.com/square/PonyDebugger) 是 由 Square 公司 开发 的 一 个 远程 
调试 工具 ， 可 以 解决 上 述 问 题 。 此 外 ， 设 备 无 需 连 接 开发 机 器 。 


它 由 两 部 分 组 成 : 


。 一 个 网 关 服 务 器 ， 可 以 在 开发 人 员 的 视图 和 应 用 间 建 立 关联 ，; 
。 一 个 连接 到 服务 器 的 客户 端 。 


网 关 服 务 器 是 用 Python 语言 编写 的 ， 你 可 以 使 用 例 11-2 中 的 命令 行 来 安装 该 服务 器 。 


fill 11-2 #48 PonyDebugger 网 关 服 务 器 




















$ curl -s V 
https://cloud.github.com/downloads/square/PonyDebugger/bootstrap-ponyd.py | V 
python - --ponyd-symlink=/usr/local/bin/ponyd -/Library/PonyDebugger @ 


$ ponyd serve --listen-interface-192.168.0.1 @ 


Q 下 载 并 安装 ponyd 守护 进程 。 
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O AMA d. EFF RAEN IP Heh, EH --listen-port 参数 配置 端口 ， 默 认 绑 定 
127.0.0.1:9000, 


在 浏览 器 中 打开 http://192.168.0.1:9000 可 以 查看 连接 的 设备 列表 。 你 应 该 可 以 看 到 类 似 图 
11-19 的 输出 结果 。 




















| 9 ® /网 PonyGateway x Wn | Gaurav | 
€ > C fi |!192.168.0.10:9000 S; f 2 PRE 
PonyGateway 


Connect a PonyDebugger client to ws://192.168.0.10:9000/device . 


No Clients Connected 


Status: Connected to gateway 























浏览 网 关 服务 器 


客户 端 API 可 以 通过 CocoaPods 集成 到 应 用 中 。Pod 名 称 为 PonyDebugger。 例 11-3 提供 了 
集成 到 应 用 中 的 示例 代码 。 


f| 11-3 PonyDebugger 一 一 客户 端 API 
#import «PonyDebugger /PonyDebugger . h» 


-(void)setupPonyDebugger ( @ 
PDDebugger *debugger = [PDDebugger defaultInstance]; @ 
[debugger connectToURL: 
[NSURL URLWithString:Q"ws://192.168.0.1:9000"]]; © 


[debugger enableRemoteLogging]; @ 

[debugger enableNetworkTrafficDebugging]; @ 
// [debugger forwardAllNetworkTraffic]; Q 
[debugger enableViewHierarchyDebugging]; @ 

[debugger enableCoreDataDebugging]; @ 








Q 添加 客户 端的 辅助 方法 。 在 任何 网 络 请 求 发 生前 ， 从 application:did FinishLaun 
chingWithOptions: 方法 中 调用 该 方法 。 

@ PDDebugger 是 要 设置 的 对 象 。 
需要 连接 的 URL， 必 须 是 ws (WebSocket 协议 )。 

O 允许 日 志 记 录 在 网 关 服 务 器 上 。 使 用 PDLog 方法 代替 NSLog 来 记录 日 志 。 

Q 启用 网 络 请 求 记 录 。 通 过 替换 NSURLConnectionDelegate 类 相关 方法 的 机 制 实现 

Q 使 用 该 方法 让 PonyDebugger 发 现 所 有 这 样 的 类 。 

O 允许 从 网 关 服务 器 查看 视图 层级 。 

O 允许 从 网 关 服 务 器 查看 Core Data 对 象 。 编 写本 书 时 尚未 支持 更 新 记录 。 


因为 使 用 PonyDebugger 会 让 应 用 的 数据 暴露 在 外 ， 所 以 建议 你 仅 在 开发 调 
试 的 安装 包 中 打开 这 项 功能 。 








一 旦 与 设备 建 并 连接， 浏览 器 就 会 展示 所 有 允许 传输 的 数据 。 


Elements 页 展示 视图 层级 。 Sh eet ds 
图 11-20 的 左边 展示 了 在 浏览 器 中 选中 节点 ， 右 边 为 应 用 中 对 应 的 高 亮 视图 。 
































Carrier > 11:28 AM = 


< Tools Pony Debugger 





| > Computed 


ss- 
(320, 480}}" alpha-"1" hidden="NO"> | Y Styles 
Y «view class="UIViewControllerwrapperView" frame="{{@, 0), | b Metrics 
(320, 480}}" alpha="1" hidden="NO"> | > Properties 1. Install desktop tools from the website 


Y «view class-"UIView 3 " "] " 
(HPChapteri7Tools PonyDebuggerViewController)" frame-"((0, | DOM Breal 2. Configure "Pony Debugger" app settings 


0), (320, 480)?" alpha-"1" hidden="NO"> | > Event List 3. Restart app 
«view class-" UllayoutGuide" frame-"((0, 431), (0, 49))" 
alpha="1" hidden-"YES"»«/view» 
«view class-" UILayoutGuide" frame="{{@, 0), 40, 64))" PonyDebugger on Github 
alpha="1" hidden-"YES"»«/view» 
<view class="UITextField" frame="{{16, 258), (288, 30))" 


alpha-"1" hidden="NO"> App Settings 
«view class="UITextFieldLabel" frame="{{7, 1), (274, PP 9 
27))" alpha="1" hidden="NO"></view> _ 


" UlITextFieldRoundedRectBackgroundViewNeue" frame=" 
{{0, 0), (288, 30))" alpha="1" hidden="NO"></view> 
</view> 
«view class-"Ullabel" frame="{{16, 74}, (288, 84))" 
alpha="1" hidden="NO"></view> 
b «view class-"UIButti frame-"((16, 298), (288, 30))" 
alpha="1" hidden="NO">..</view> 
b «view class="UIButt frame="{{16, 218), (288, 30}}" 
alpha-"1" hidden="NO">..</view> 
b «view class-"UIButton" frame="{{16, 178), (288, 30))" 
alpha-"1" hidden="NO">..</view> 
</view> 


cf 
um | a GG * SG 


Chapters Debug Log 





To use pony debugger: 





Make Network Request 














11-20; PonyDebugger 一 一 视图 层级 
在 图 11-21 中 可 以 找到 网 络 页 ， 其 中 列 出 了 应 用 启动 后 按时 间 先 后 顺序 排列 的 所 有 网 络 请 
求 。 记 住 以 下 重要 的 数据 : 


。 连接 对 应 的 不 同 域名 
。 每 个 响应 消耗 的 时 间 
。 每 个 请 求 传输 的 数据 大 小 


主意， 因为 这 个 工具 拦截 的 是 NSURLConnection 委托 回调 ， 所 以 它 只 能 处 理 http 和 https 
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请 求 。 如 果 你 使 用 原始 套 接 字 或 其 他 连接 方式 ， 那 PonyDebugger 并 不 能 拦截 到 它们 。 

在 网 络 页 上 点 击 一 个 请 求 就 能 查看 该 请 求 的 详细 数据 ， 与 图 11-22 所 示 类 似 。 你 可 以 看 到 
请 求 和 响应 头 、 响 应 数据 ， 以 及 发 送 和 接收 的 cookie。 就 POST 或 PUT 请 求 而 言 ， 你 还 可 以 
查看 请 求 体 。 














Path Met... Tex:” Type Initiator dee ur Timeline | 5.97s! B895s 1194s 149zs| 
国 Weenie com |S" oam = Other iran som © 

Lj E SE i um (uis OT prs ed e 

FewemE 00 0o a e 





3 requests | 120.91KB transferred 


rf v GS [An | Documents Stylesheets Images Scripts XHR Fonts WebSockets Other 














图 11-21; PonyDebugger 一 一 网 络 : 查看 所 有 请 求 



























Headers | Preview Response Cookies 


www.google.com Request URL: http://www. google. com/ 
www.google.com Request Method: GET 


Status Code: (9200 no error 














www.msn.com 


v Request Headers view source 
www.msn.com Accept: 六 /x 
[ www.m10v.com Accept-Encoding: gzip, deflate 
www.m10v.com Accept-Language: en-us 
v Response Headers view source 


Accept-Ranges: none 

Alternate-Protocol: 80:quic,p-0 

Cache-Control: private, max-age-0 

Content-Type: text/html; charset=IS0-8859-1 

Date: Sun, 07 Jun 2015 19:28:56 GMT 

Expires: -1 

Server: gws 

Set-Cookie: PREF=ID=86a5158e7a3a50a1 : U=1458adcdd4996f87 : FF=0: TM=14337 
04070:LM-1433705336:S-lNEeJ5WHOFLLZUz7; expires=Tue, 06-Jun-2017 1 
9:28:56 GMT; path=/; domain-.google.com 

Transfer-Encoding: Identity 

Vary: Accept-Encoding 

X-Frame-Options: SAMEORIGIN 

X-XSS-Protection: 1; mode-block 





3 requests | 120.91KB transferre 
| MES | e | S | €o | Documents Stylesheets Images Scripts XHR Fonts WebSockets Other 











图 11-22; PonyDebugger 一 一 网 络 : 调试 单个 请 求 
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Console 页 展示 通过 调用 NSLog 方法 记录 的 所 有 消息 (如 图 11-23 tas). NSLog 不 仅 会 在 调 
试 期 间 将 数据 消息 写 入 Xcode 控制 台 ， 还 会 写 到 苹果 日 志 系统 中 。 因 为 需要 进程 间 通 信 ， 
所 以 苹果 日 志 系 统 的 调用 开销 很 大 。 除 此 之 外 ， 在 非 调试 版 本 的 应 用 ， 包 括 从 App Store 
安装 应 用 时 ， 通 过 Xcode 仍然 可 以 记录 日 志 。 记 录 私 钥 、 密 码 等 敏感 信息 会 引发 安全 问 
题 。 留 意 日 志 中 的 此 类 数据 。 












ea one ae) A 





[V] prepareForSegue[HPCVC] destination: ident-segue ch12 tools cls=HPChapter17ToolsTableViewController 
[V] prepareForSegue[HPCVC] destination: ident-segue ch03 idents cls=HPChapter@3ViewController 
[D] [enter] createStrongPhoto 

[D] Strong Photo: «HPPhoto: 0x174443cc0» 

[D] [exit] createStrongPhoto 

[V] HPVPhoto dealloc-ed 

[D] [enter] createWeakPhoto 

[V] HPVPhoto dealloc-ed 

[D] Weak Photo: (null) 

[D] [exit] createWeakPhoto 

[D] [enter] createStrongToWeakPhoto 

[D] Strong Photo: «HPPhoto: 0x17044b670» 

[D] Weak Photo: «HPPhoto: 0x17044b670» 

[D] [exit] createStrongToWeakPhoto 

[V] HPVPhoto dealloc-ed 











图 11-23; PonyDebugger 





控制 台 : 远程 日 志 


使 用 PonyDebugger 的 网 关 服 务 器 来 监控 和 分 析 日 志 非 常 简 单 。 并 且 ， 因 为 它 是 开源 的 ， 
所 以 你 可 以 更 新 代码 来 拦截 套 接 字 ， 以 便 分 析 数 据 以 进行 自动 化 测试 。 


11.5 Charles 


XK72 公司 的 Charles 代理 服务 器 是 一 个 监控 、 管 理 http 和 https 请 求 的 工具 。 

我 们 在 7.3.3 节 中 曾 简单 介绍 过 该 工具 ， 此 节 将 继续 深入 探讨 。 

Charles 可 以 在 被 称 为 一 个 会 话 的 过 程 中 记录 下 所 有 的 请 求 。 因 此 ， 一 个 会 话 可 以 有 多 个 请 
求 。 通 过 Proxy 一 Start Recording 或 Proxy 一 Stop Recording， 你 可 以 启动 或 关闭 记录 。 


每 个 会 话 可 以 具备 一 批 URL 的 白 名 单 和 黑 名 单 ， 可 通过 Proxy 一 Recording Settings 的 
Include 或 Exclude 页 面 设置 这 些 名 单 ， 如 图 11-24 所 示 。 当 白 名 单 为 室 ， 或 一 个 URL 在 白 
名 单 且 不 在 黑 名 单 中 时 ， 这 个 URL 将 出 现在 会 话 中 。 
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» Include Locations 


Recording Settings 


Coninn. Se «oes lane) 


Recording Settings 


Options | Include © 








excluded. 


Only requests that match one of the locations below will be recorded. If 
this list is empty, all requests will be recorded unless otherwise 


y Exclude Locations 


Requests that 








Xu 





match one of the locations below will not be recorded. 


Add 


(Remove ) 





(Cancel ) GOK 


(import) ( Export ) 


— 
Gare) QD 








图 11-24; Charles ——URL 白 名 单 和 黑 名 单 


可 以 按照 {时 间 先 后 顺 


序 查看 会 
径 为 子 结 点 的 属性 结构 查看 ， 如 





图 BEAR 








11-26 PR. dE 





T A 





e 用 。 


话 中 的 URL， 如 图 11-25 所 示 ， 或 者 根据 域名 为 根 结 点 、 路 
要 检查 请 求 的 顺序 ， 那 么 第 一 种 格式 
若 想 要 从 某 个 特定 的 域名 或 路 径 调试 请 求 ， 那 么 第 二 种 格式 更 有 利 。 





D al wm EM 


Charles 3.10.1 - Session 1 * 


ore GQ/v x5 


























RC Method Host 
CONNECT www.google.com 


www.m10v.com 
www.m10v.com 
www.m10v.com 
www.m10v.com 





www.m10v.com 





Path Start Duration 


17:41:23 59335 ms 


pem -includes/css/dashicons.min.css?ve... 17:41:25 1355 ms 
|wp-content/plugins/jetpack/css/jetpa... 17:41:25 473 ms 
/wp-includes/css/admin-bar.min.css?v... 17:41:25 426 ms 
[wp-includes/js/jquery/jquery.js?vers... 17:41:25 ^ 585 ms 








/wp-content/themes /nest/scripts/nest-... 17:41:25 


377 ms 


Size Status 


4.57 KB Complete 


Info 





29.55 KB Complete 
11.65 KB Complete 
5.37 KB Complete 
34.17 KB Complete 
1.81 KB Complete 


CICI 





Filter: 


O Focussed 





Í Overview "Reqüest^| Response 





Summary 





Chart 


Notes | 








GET / HTTP/1.1 
Host www.m10v.com 


Accept-Language en-US,en;q=0.5 
Accept-Encoding gzip, deflate 
DNT 1 
Cookie _utma=111043097. 
Connection keep-alive 


User-Agent Mozilla/5.0 (Macintosh; Intel Mac OS X 10.11; rv:43.0) Gecko/20100101 Firefox/43.0 
Accept text/html,application/xhtml4-xml,application/xml;q » 0.9,* /*:q 0.8 


1927304523.1417675940.1453335103.1453504788.179; 


Cookies | Raw 


..utmz-111043097.1452145110.173.4.utmcsr- google... 

















图 11-25; Charles 一 一 会 话 中 的 URL 按时 间 先 后 
使 用 树 形 格式 可 以 提供 一 个 全 景 


中 非常 有 用 。 
在 Overview 页 











顺序 排列 
如 图 11-26 所 示 。 


I^ 
网 ， 





4 请 求 〈 请 求 总 数 、 


完成 和 失败 的 崩溃 、 





此 外 ， 该 格式 在 以 下 的 调试 任务 


i 可 以 查看 一 个 域名 及 域名 下 的 子路 径 相关 的 所 有 请 求 。 你 可 以 更 深入 
地 研究 http 和 https 请 


求 。 图 11-27 展示 了 以 下 元 素 : 


服务 器 连接 状态 ， 以 及 SSL 握手 次 数 ) 
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4 整体 用 时 、DNS 解析 用 时 、 创 建 到 主机 的 连接 用 时 , 以 及 SSL 握手 时 间 (如 果 可 以 的 话 ) 
€ 请 求 (针对 Post Put 方法 ) 以 及 以 KBps 为 单位 的 响应 速度 
4 请 求 及 以 KB 为 单位 的 响应 大 小 


。 对 于 时 间 、 速 度 和 数据 大 小 ， 可 以 查看 总 数 、 最 小 值 、 最 大 值 以 及 平均 值 等 详细 数据 ， 
进而 可 以 深入 了 解 应 用 产生 的 网 络 请 求 。 























e^o Charles 3.10.1 - Session 1 * 
* wm ABA oe FO BSH Xe 
> da hups://www.google.com Name | Vae oo E- 
> 县 https://www.charlesproxy.com Host http://www.m10v.com 
> © http://control.charles Path / 
了 ® http://www.m10v.com Notes 
> im wp-includes * Requests 15 
> W wp-content Completed 15 
Q9 «default» Incomplete 0 
> @ http:/150.wp.com Failed 0 
> @ hup://s.gravatar.com Blocked 0 
> @ http://www.codeproject.com DNS 1 
Connects 11 
SSL Handshakes 0 
^| v Timing 
Start 1/23/16 17:41:23 
End 1/23/16 17:41:29 
Timespan 6.09 sec 
Requests / sec 2.46 
» Duration 12.85 sec 
b DNS 118 ms 
> Connect 1.69 sec 
> SSL Handshake  - 
> Latency 5.26 sec 
> Speed 120.04 KB/s 
> RequestSpeed 19,535.16 KB/s 
» Response Speed 265.16 KB/s 14 
| y Size M 























图 11-26; Charles 一 一 会 话 中 的 URL 按 树 形 结构 排列 





eoe J : Charles 3.10.1 - Session 1 * 
Dhang ero a/v Xe 


> & https://www.google.com 
> ®& https://www.charlesproxy.com 
> @ http://control.charles 














v (http: //www.m10v.com | Completed 
* im wp-includes Incomplete 
v id Css Failed 
© dashicons.min.css?verz 4.4.1 Blocked 
© admin-bar.min.css?ver- 4.4.1 DNS 
> djs Connects 


> ji wp-content SSL Handshakes 
@ «default» || Y Timing 
> © hup://s0.wp.com | Start 1/23/16 17:41:25 
> @ http://s.gravatar.com Si End 1/23/16 17:41:27 
d [5] http://www.codeproject.com Timespan 1.36 sec 
Requests /sec 3.69 
Y Duration 3.27 sec 
Min 426 ms 
Mean 654 ms 
Max 1.36 sec 
Total 3.27 sec 
» DNS = 


Y Connect 706 ms 
Min 32ms 
Mean 141 ms 
Max 313 ms 


























图 11-27; Charles 一 一 指定 子路 径 的 请 求 和 响应 概览 
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* Summary 页 提供 了 HTTP 的 响应 值 、 内 容 类 型 (MIME 类 型 )、 包 头 大 小 、 请 求 体 大 小 ， 
以 及 每 个 指定 子路 径 下 的 URL 的 请 求 完成 时 间 ， 如 图 11-28 所 示 。 





ON ma EL Á 


Charles 3.10.1 - Session 1 * 


et e@ Sv X# 








Overview Summarys) Chart 








> & https://www.google.com # Resource Host Code Mime Type Header Body Time | 
> 全 hups://www.charlesproxy.com 1 © dashicons.min.css?ver=4.4.1  www.m10v.com 200 text/css 4108 27.88KB 1.45 
> @ hup://control.charles 2 © admin-bar.min.css?ver=4.4.1 www.mlOv.com 200 text/css 409B 3.71KB 426ms 
了 @ http://www.m10v.com 3S jquery.jstver=1.11.3 www.mlOv.com 200 application... 427B 32.50 KB 585ms | 
vw Wm wp-includes 4 © admin-bar.min,js?ver=4.4.1  www.mlOv.com 200 application... 425 B 2.42 KB — 452ms 
v là css 5 © wp-embed.min.js?ver-4.4.1 — www.mlOv.com 200 application... 423 8 7668 453ms | 
@ dashicons.min.css?ver=4.4.1 Total 2.04 KB 67.26KB 3.35 
© admin-bar.min.css?ver=4.4.1 Grand Total 69.31 KB 
"das Duration 14s 


> B wp-content 
(9 <default> 

> @ http://s0.wp.com 

> @ hup://s.gravatar.com 


> @ http://www.codeproject.com 























- j 
图 11-28: Charles 一 一 指定 子路 径 下 所 有 请 求 的 摘要 
。 Chart 页 提供 了 图 形 化 分 析 来 了 解 时 间 线 、 请 求 大 小 、 持 续 时 间 和 内 容 类 型 。 
(D 当 请 求 发 起 时 ，Timeline 子 页 会 展示 建立 连接 、 等 待 响应 以 及 接收 响应 (从 收 到 第 
一 个 字 节 开始 算 起 ) 相应 的 时 间 。 这 些 内 容 体 现在 图 11-29 中 ， 用 于 识别 那些 连接 
和 响应 缓慢 的 URL 和 域名 。 























eoe Charles 3.10.1 - Session 1 * 
> am AeA eo COBY xXx 
& https: / /www.google.com # Resource Timeline | 
Â https://www.charlesproxy.com 1 © dashicons.min.c... MENS UNN.:. 
© http://control.charles 2 @ admin-bar.min.... EE +26 
@ htp://www.mlovcom 3 TE jaueryjsiver-1... I 5 35; 
4 1* admin-bar.minj… | Ee 
5 CT wp-embed.min.... ESE 
© dashicons.min.css?ver-4.4.1 Total Duration 14s 
© admin-bar.min.css?vere 4.4.1 
> m js 
> Wa wp-content 
Q «default» 
@ http: //s0.wp.com 
@ http://s.gravatar.com ii 
@ http: //www.codeproject.com 
Sizes | Duration Types | 























图 11-29; Charles 一 一 指定 子路 径 下 请 求 的 相对 时 间 轴 
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(2) Size 子 页 展示 了 响应 大 小 的 一 个 柱 形 图 ， 按 数据 量 从 大 到 小 排序 〈 见 图 11-30), 1% 
数据 可 用 于 优化 响应 大 小 、 确 认 服 务 器 扩容 需要 ， 或 增加 缓存 内 容 的 边缘 服务 器 。 





eae Charles 3.10.1 - Session 1 * 
» @PmAeAe C€@ a/y X# 


> & hups://www.google.com # Resource Size | 

















> à https://www.charlesproxy.com 1& jqueryjsver= 1.1... | 2 02 «£ 
> Q9 hup://control.charles 2 © dashicons.min.css?.. EE... 
v @ http://www.m10v.com 3 © admin-bar.min.css... EN 4 10 /¢ 
v Mm wp-includes 4 & admin-bar.min.js?... 
v lr css 5 © wp-embed.min.js?... M16 Ke 
© dashicons.min.css?ver=4.4.1 Total Size 69.31 KB 
© admin-bar.min.css?ver-4.4.1 
> mis 
> lii wp-content 
(9 «default» 


> © http://s0.wp.com 
> @ hup://s.gravatar.com 
> @ hup://www.codeproject.com 


[ Timeline “Sizes Duration Types | 




















图 11-30; Charles 一 一 指定 子路 径 下 请 求 的 响应 数据 的 大 小 


(3) Duration 子 页 ( 见 图 11-31) 提供 了 每 个 请 求 的 响应 时 间 数 据 ， 依 照 用 时 长 短 降序 排 
列 。 该 数据 可 用 于 发 现 响应 缓慢 的 请 求 和 URL， 并 依 此 优化 服务 器 。 


ece Charles 3.10.1 - Session 1 * 
Dhang erro a/y xt 


> & hups://www.google.com # Resource Duration | 
» 全 https://www.charlesproxy.com 1 O dashicons. min. .. 
> @ hup://control.charles 2 © jaueryjsjver-... -= 
v @ hup://www.m10v.com 3 '* wp-embed.mi.. MEE... 
v P wp-includes 4 Ù% admin-bar.mi.. i 2 
v lr css 5 © admin-bar.mi.. LT 326; 
© dashicons.min.css?ver- 4.4.1 
© admin-bar.min.css?ver-4.4.1 
> W js 
> W wp-content 
(9 «default» 
> © htp://s0.wp.com 
> @ hup://s.gravatar.com 
> @ http://www.codeproject.com 




















{ Timeline | Sizes "Duration | Types | 























& 11-31; Charles 一 一 指定 子路 径 下 请 求 的 响应 时 间 








Charles 是 一 个 从 应 用 全 面 监控 网 络 的 工具 。 它 不 仅 可 以 标识 出 所 有 发 出 的 请 求 ， 还 可 以 深 
入 研究 单个 请 求 以 明确 任何 时 间 和 数据 大 小 的 瓶颈 。 


11.6 ”小 结 


调试 工具 可 以 为 开发 人 员 提 供 帮 助 。 使 用 内 租 的 代码 来 测量 内 存 、CPU 使 用 率 这 样 的 参 
数 ， 也 许 并 不 能 得 到 真实 的 结果 ， 因 为 这 部 分 代码 也 消耗 了 资源 。 调 试 工具 为 测量 性 能 饼 
各 方面 因素 提供 了 非 侵 入 式 的 选择 。 

苹果 公司 的 Accessibility Inspector 可 以 测试 应 用 无 障碍 方面 相关 的 属性 。 

Instruments he 内 存 使 用 率 等 性 能 参数 、 检 查 可 能 存在 的 内 存 泄漏 、 查 看 视图 层级 关 
系 及 其 复杂 度 ， 等 等 。 相 关 图 表 的 波动 都 应 当 予 以 关注 。 同样 ， 一 个 逐渐 增长 的 内 存 使 用 
图 表明 存在 潜在 的 内 存 泄 漏 。 
PonyDebugger 可 以 监测 和 调试 视图 层级 ， 而 无 需 暂 停 运 行 中 的 应 用 。 它 同样 可 以 监控 
Core Data 的 使 用 。 

Charles 可 以 在 应 用 外 部 监控 网 络 活动 ， 尤 其 是 在 不 侵入 应 用 代码 的 情况 下 测试 各 种 场景 。 


因为 引入 了 许多 手动 操作 步 又， 所 以 使 用 调试 工具 会 减 慢 性 能 的 整体 测试 速度 ， 但 你 还 是 
应 当 定 期 使 用 它们 来 保持 应 用 的 稳定 。 
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埋 点 与 分 析 





应 用 开发 初期 的 优化 依据 通常 基于 最 佳 实践 、 指 南 ， 以 及 从 开发 者 机 器 或 实验 室 收集 的 数 
据 。 然 而 ， 这 些 仅 是 数据 分 析 的 初始 集合 。 

直到 应 用 发 布 后 ， 我 们 才能 收集 多 个 设备 和 地 理 位 置 的 真实 数据 ， 帮 助 明确 用 户 的 使 用 模 
式 和 需要 调整 的 各 种 场景 。 

在 第 一 章 中 ， 我 们 探讨 了 用 于 测量 和 调整 应 用 的 参数 ， 其 中 包括 以 下 内 容 : 

。 内 存 使 用 情况 

。 响应 时 间 

。 网 络 使 用 情况 

。 本 地 存储 

我 们 已 经 在 前 文中 讨论 了 改善 用 户 体验 的 各 种 策略 ， 并 明确 了 让 应 用 有 更 高 性 能 的 特定 方 
法 ， 现 在 是 时 候 讨 论 如 何 从 真实 用 户 那儿 收集 数据 、 分 析 应 用 使 用 情况 、 发 现任 何 性 能 瓶 
颈 、 提 供 修复 和 更 新 版 本 ， 从 而 让 用 户 对 应 用 更 加 满意 。 

本 章 讨 论 的 领域 包括 : 如 何 分 析 收 集 到 的 生产 数据 ， 用 以 发 现 应 用 的 使 用 趋势 、 用 户 行 
为 ， 并 通过 埋 点 、 分 析 和 真实 用 户 监控 对 应 用 进行 改进 和 优化 。 


12.1 词汇 
在 展开 具体 内 容 前 ， 我 们 先 来 了 解 一 下 本 章 讨 论 中 将 会 用 到 的 词汇 。 


。 属性 
需要 获取 具体 值 的 参数 ， 如 应 用 版 本 、 系 统 版 本 、 位 置 、 语 言 、 使 用 内 存 等 。 
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事件 

在 应 用 中 发 生 的 任何 行为 ， 无 论 是 由 用 户 还 是 由 应 用 本 身 触发 。 

用 户 触 发 的 事件 包括 登录 、 观 看 视频 等 。 应 用 触发 的 事件 包括 冷 启 动 、 后 台 同 步 、 下 载 
邮件 等 。 
一 个 事件 是 一 系列 属性 的 集合 ， 这 些 属性 包括 系统 版 本 、 设 备 型 号 、 应 用 冷 局 动 的 用 
时 、 后 台 同 步 传输 的 数据 量 、 下 载 邮 件 占用 的 内 存 ， 等 等 。 

漏斗 

用 来 测量 用 户 如 何在 一 系列 事件 中 切换 的 工具 。 

一 个 漏斗 可 以 用 来 发 现 使 用 模式 以 及 常见 任务 或 应 用 退出 的 位 置 。 

埋 点 

监控 或 测量 性 能 水 平和 诊断 错误 的 一 种 能 力 。 在 应 用 开发 领域 ， 它 指 的 是 向 服务 器 发 送 
对 应 的 事件 以 便 分 析 。 

埋 点 源码 

在 应 用 中 注入 埋 点 代码 。 

注入 的 代码 可 以 由 开发 人 员 手 写 或 在 编译 时 使 用 工具 自动 生成 ， 你 也 可 以 使 用 方法 替换 
在 运行 时 监视 。 

要 想 在 用 户 的 移动 设备 上 分 析 生 产 环境 中 的 应 用 ,添加 埋 点 代码 是 唯一 的 方法 。 你 无 
法 使 用 Instruments 这 样 的 工具 ， 因 为 移动 设备 不 能 连接 到 开发 人 员 的 机 器 。 男 一 方面 ， 
你 也 无 法 使 用 PonyDebugger, Charles 这 样 的 工具 ， 因 为 移动 设备 可 能 不 和 工程 师 团 队 
在 同一 个 网 络 、 被 防火 墙 阻隔 ， 或 根本 无 网 络 连 接 。 

分 析 

在 数据 中 发 现 和 传达 有 意义 的 模式 。 

在 应 用 开发 领域 ,分 析 数 据 来 源 于 应 用 埋 点 。 

同类 群 组 

特定 时 间 内 拥有 共同 特征 的 一 组 用 户 。 

例如 ， 同 性 别 的 用 户 、 在 同一 座 城市 的 用 户 ， 以 及 在 特定 日 期 开始 使 用 应 用 的 用 户 。 








































































































同类 群 组 分 析 
对 根据 同类 群 组 特征 区 分 的 数据 进行 分 析 。 
归 因 


将 应 用 的 销售 功劳 和 转化 功劳 分 配给 转化 路 径 中 的 接触 点 。 
用 户 可 能 有 多 个 选项 来 开始 购买 或 完成 任务 。 归 因 模 型 规定 了 哪些 选项 将 获得 积分 ， 并 
因此 获得 在 广告 活动 中 花费 的 资金 份额 。 

真实 用 户 监控 

被 动 监控 技术 ， 用 于 记录 用 户 与 应 用 的 所 有 交互 ， 将 其 发 送 到 服务 器 ， 并 帮助 监控 使 用 
情况 、 趋 势 和 出 现 的 问题 。 
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12.2 EA 
埋 点 有 一 些 可 行 的 技术 ， 甚 中 包括 二 进 制 代码 重 写 ， 以 及 随处 、 即 时 、 链 接 时 和 源码 中 的 
Hs. 


我 们 主要 关注 源码 层面 的 埋 点 。 特 别 是 在 应 用 导航 和 资源 使 用 方面 ， 它 可 以 帮助 我 们 分 析 
应 用 。 


我 们 在 第 1 章 中 介绍 过 埋 点 。 在 例 1-2 中 ， 我 们 创建 了 HPInstrumentation 类 以 记录 事件 ， 
事件 用 于 标识 一 种 期 望 行为 的 发 生 或 未 发 生 。 


例如 ， 如 果 我 们 想 跟踪 丝 存 组 件 的 性 能 ， 我 们 可 以 在 它 获取 缓存 对 象 的 方法 中 记录 一 个 缓 
存 的 命中 事件 或 未 命中 的 事件 。 


类 似 地 ， 针 对 电 商 应 用 ， 我 们 可 能 想 要 确定 用 户 在 浏览 哪个 页 面 、 哪 个 产品 ， 以 及 为 每 个 
产品 花 了 多 长 时 间 等 信息 。 


从 概念 上 来 说 ， 埋 点 与 日 志 并 没有 区 别 ， 只 是 埋 点 的 目的 是 将 数据 储存 在 服务 器 ， 从 而 达 
到 持久 化 ， 并 将 其 用 于 分 析 ， 包 括 离线 批 处 理 任务 或 实时 计算 。 


接 下 来 我 们 将 了 解 埋 点 的 三 个 阶段 一 一 规划 、 实 现 和 发 布 。 












































12.2.1 规划 
埋 点 需要 深入 规划 。 规 划 的 第 一 步 是 就 使 用 第 三 方 库 还 是 自己 实现 作出 决定 。 
刚 入 门 时 ， 选 择 第 三 方 库 是 比较 稳妥 的 。 随 着 数据 、 容 量 以 及 功能 需求 增长 ， 你 可 以 以 后 
再 构建 自己 的 库 。 
以 下 是 部 分 常用 的 iOS 第 三 方 统计 库 。 
。 Flurry (http://developer.yahoo.com/flurry) 
最 受 欢迎 的 SDK 之 一 。 全 人 免费 版 本 。 文 持 WatchKit。 


e Mixpanel (https://mixpanel.com) 
支持 更 多 的 复杂 功能 ， 比 如 AB 测试 ， 无 需 编写 代码 和 调查 。 免 费 版 本 每 月 可 以 上 报 
25 000 个 数据 点 。 








e Appsee (https://www.appsee.com) 
支持 交互 热力 图 (如 图 12-1 所 示 ) 以 及 录制 应 用 使 用 的 视频 。 无 免费 版 本 。 
e Upsight (http://www.upsight.com/analytics) 
支持 归 因 和 同类 群 组 分 析 。 有 免费 版 和 付费 版 。 
e Google Analytics (http://bit.ly/google-analytics-sdk-ios) 
从 社交 和 电 商 的 角度 支持 更 多 功能 。 每 名 用 户 每 天 最 多 有 200 000 次 免费 点 击 事 件 ， 每 
个 会 话 最 多 有 500 次 事件 上 报 。 上 述 限制 同样 适用 于 高 级 账户 。 
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图 12-1. 在 一 个 视图 控制 器 中 的 用 户 交 互 的 热力 图 (SARE Appsee) 


市 面 上 还 有 其 他 的 SDK， 但 是 上 述 SDK 是 很 好 的 起 步 选择 。 
分 析 引 擎 中 应 当 包含 以 下 的 重要 功能 (假设 已 支持 通过 名 称 记录 事件 ， 而 且 总 是 可 以 获取 
设备 信息 、 地 区 、 时 区 、 位 置 、 网 络 /运营 商 等 基本 数据 )。 
。 可 扩展 事件 

可 以 对 任意 事件 添加 自 定义 参数 或 内 容 。 
。 ARAL bE 

使 用 可 扩展 事件 的 参数 ， 过 滤 和 分 析 其 中 特定 的 值 。 
。 事件 计时 

可 以 获取 事件 的 持续 时 间 。 
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。 记录 所 有 页 面 视 图 
可 以 分 析 视 图 控制 器 的 展示 和 消失 。 

。 用 户 
可 以 设置 用 户 ， 从 而 跟踪 匿名 用 户 ， 记 录 事 件 。 

。 交易 
可 以 提供 交易 的 货币 价值 ， 常 用 于 电 商 或 使 用 内 购 的 应 用 。 

。 A/B 测试 
可 以 进行 A/B 测试 ， 监 控 用 户 行为 。 

。 实时 数据 
可 以 获取 实时 或 近 实 时 数据 。 根 据 不 同 的 应 用 ， 可 接受 的 事件 上 报 延 迟 时 间 可 以 从 几 分 
钟 到 几 小 时 。 

















。 安全 
可 以 监控 安全 ， 监 控 与 服务 器 通信 的 安全 程度 ， 以 及 服务 器 上 的 数据 的 安全 程度 。 
。 会 话 回放 





可 以 录制 视频 ， 之 后 回放 。 这 样 可 以 近 距 离 监控 应 用 的 使 用 情况 ， 更 好 地 发 现 出 错 场 
景 ， 更 快 地 对 其 进行 修复 。 


谨慎 使 用 这 项 功能 ， 因 为 这 可 能 要 考虑 到 隐私 和 法 律 方面 的 影响 。 最 佳 的 使 
用 场合 是 在 私有 用 户 研 究 会 话 。 

如 果 你 打算 实现 这 项 功能 ， 确 保 使 用 突出 的 提示 界面 告知 用 户 关 于 录制 的 行 
为 。 此 外 ， 在 录制 前 还 需要 明确 获得 用 户 授权 。 


























。 热力 图 
可 以 产生 热力 图 来 确认 应 用 的 热点 和 盲点 。 
。 JAB 
可 以 跟踪 点 击 以 及 应 用 安装 的 归 因 。 
。 活动 
支持 有 组 织 或 自发 的 活动 。 常 用 于 市 场 分 析 。 
。 漏斗 
使 用 事件 流 定义 漏斗 。 


。 原始 事件 
在 复杂 的 处 理 过 程 中 使 用 源 数 据 是 锦上添花 的 一 件 事 。 这 可 能 会 在 企业 内 部 的 方案 中 
使 用 。 












































12.2.2 ”实现 
一 旦 确定 埋 点 方案 ， 下 一 步 就 是 配置 。 为 了 进行 应 用 埋 点 ， 你 需要 进行 如 下 设置 。 
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确定 指标 

这 个 过 程 需 要 产品 、 营 销 和 工程 师 团队 的 合作 。 产 品 经 理 需 要 用 户 体验 统计 ， 营 销 团 队 
感 兴趣 的 是 应 用 使 用 情况 和 用 户 感 兴趣 的 部 分 ， 而 工程 团队 希望 了 解 应 用 的 性 能 。 

定义 事件 

定义 事件 名 称 和 相关 内 容 来 支持 关键 的 绩效 指标 。 

如 果 工 程 师 团队 需要 内 存 使 用 的 均值 和 峰值 ， 那 么 事件 (我们 称 之 为 Heap Size) 应 该 包 
括 已 使 用 内 存 的 数据 、 空 闲 内 存 (回忆 例 2-40 中 的 代码 )， 以 及 应 用 启动 和 运行 的 时 间 。 
另 一 个 类 似 的 例子 是 ， 如 果 产 品 经 理想 了 解 新 的 自动 填 表 功能 是 否 受到 好 评 ， 那 么 他 不 
仅 需 要 使 用 该 功能 的 事件 ， 还 需要 了 解 自动 填 表 能 节约 多 少时 间 用 于 评价 效率 。 

你 可 能 需要 记 时 事件 。 例 如 ， 在 一 个 新 闻 应 用 中 ， 可 能 需要 记录 用 户 在 其 篇 新 闻 或 特定 
类 别 中 花费 了 多 少时 间 。 根 据 使 用 的 SDK， 你 可 以 使 用 内 置 的 计时 事件 。! 否则 的 话 ， 
你 需要 自行 实现 一 个 时 间 监 控 功 能 。 

编写 代码 

一 旦 明确 所 有 事件 ， 并 且 确 定 每 个 事件 何 时 被 调用 ， 那 就 可 以 编写 代码 了 。 

良好 的 实践 经 验 是 创建 一 个 类 ， 其 中 包含 应 用 需要 的 所 有 埋 点 方法 。 例 12-1 包含 了 应 
用 需要 的 一 些 通 用 方法 。 可 以 在 此 基础 上 ， 根 据 需 要 逐步 增加 。 

例 12-1 埋 点 类 一 一 方法 列表 


@interface HPInstrumentation 

































































*(void)logEvent:(NSString *)name params:(NSDictionary *)params; @ 
*(void)startTimerForEvent:(NSString *)name params:(NSDictionary *)params; 
*(void)endTimerForEvent:(NSString *)name params:(NSDictionary *)params; @ 
*(void)logViewControllerDidAppear:(UIViewController *)vc; © 


*(void)setLocation:(CLLocation *)location; @ 
*(void)setUserId:(NSString *)userId; @ 


*(void)logError:(NSString *)name 
message:(NSString *)message 
exception: (NSException *)e; © 


*(void)setMinimumTimeBetweenSessions:(NSInterval)interval; @ 
@end 


Q 必要 方法 ， 记 录 一 般 事件 。 
@ 记录 计时 事件 。 
© ic Pos BA E s tll 





ü 

















E1: Flurry SDK 支持 计时 事件 。 参 见 Flurry 开发 者 文档 中 的 “Capture Event Duration". (http://yhoo.it/IcTSpNL) . 
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O 为 后 续 事 件 设置 位 置信 息 。 

因为 后 续 事件 设置 用 户 ID ， 常 用 于 记录 登录 后 的 行为 。 

Q 记录 错误 的 特殊 事件 。 

@ 计算 会 话 次 数 的 特殊 处 理 ， 影 响 日 常 活跃 用 户 的 计数 。 应 当 将 应 用 切 到 后 台 再 切 回 
前 台 算 作 一 次 新 的 会 话 吗 ?或 者 这 两 个 行为 中 间 还 需要 一 个 最 小 间隔 时 间 ? 

你 可 以 使 用 Aspects 的 CocoaPod 库 (https://github.com/steipete/Aspects) 来 设置 常用 跟 

踪 事 件 。 比 如 ， 如 果 要 跟踪 所 有 UIViewController 的 viewDidAppear: 方法 ， 你 可 以 使 

用 例 12-2 中 的 代码 。 


f| 12-2 使 用 Aspects 的 CocoaPod 库 进 行 无 感知 方法 跟踪 


[UIViewController aspect_hookSelector:@selector(viewDidAppear:) @ 
withOptions:AspectPositionAfter @ 
usingBlock:^(id«AspectInfo» info, BOOL animated) { © 
NSDictionary *eventParams = @{ 
Q"ViewControllerClass": [info.instance class] 
no 
[HPInstrumentation logEvent:Q"viewDidAppear" 
withParameters:eventParams]; © 
) error:NULL]; 


O 添加 一 个 钓 子 方法 到 类 UIViewController 的 viewDidAppear: 方法 上 。 
O 钩子 方法 必须 在 原始 方法 调用 后 生效 。 
© 实现 钧 子 方法 。 参 数 是 id<AspectInfo>， 提 供 了 调用 块 的 上 下 文 对 象 ， 以 及 原始 方 
法 (在 此 为 viewDidAppear:) 需要 的 参数 。 

Q 设置 被 记录 事件 的 参数 。 
O 记录 日 志 。 
根据 特殊 需求 还 可 以 定义 更 多 的 方法 ， 例 如， 可 以 定义 描述 购买 或 退 款 等 交易 过 程 
的 方法 。” 

。 验证 
记 住 ， 在 发 布 前 要 先 验证 。 测 试 不 仅仅 要 考虑 正确 性 ， 同 时 也 要 考虑 规模 。 确 保 依 赖 的 
第 三 方 服务 有 足够 的 容量 不 会 被 应 用 事件 的 流量 压 垮 。 
























































12.2.3 Zz 

最 后 一 步 就 是 部 署 了 。 它 涉及 发 布 服务 器 到 生产 (如果 使 用 企业 内 的 解决 方案 )， 以 及 发 
布 应 用 到 App Store. 

使 用 采集 到 的 数据 生成 报告 ， 明 确 模 式 和 趋势 并 不 是 埋 点 的 任务 ， 但 这 在 分 析 阶 段 十 分 重 
要 ， 我 们 将 在 下 文中 展开 讨论 。 








iE 2: Google Analytics 的 API 支持 跟踪 交易 事件 (https://goo.gls/ibvY Yt) 。 
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12.3 分 析 


分 析 是 发 现 和 呈现 有 意义 的 数据 模式 ， 通 常 以 数据 可 视 化 的 方式 呈现 ， 比 如 图 表 、 地 理 位 
置 图 、 热 力图 等 。 


在 应 用 开发 的 范 暑 ,分 析 使 用 埋 点 事件 产生 的 数据 来 展示 有 利于 实现 目标 规划 的 洞 见 。 

分 析 方 案 通常 会 处 理 一 部 分 数据 来 提供 高 层次 的 趋势 。 百 分 比 采 样 的 过 程 可 以 在 客户 端 或 
服务 端 完 成 。 如 果 在 客户 端 完成 ， 就 会 是 所 有 事件 的 一 小 部 分 上 报 给 服务 器 。 如 果 在 服务 
端 完成 ， 客 户 端 则 会 上 传 所 有 的 数据 ， 但 服务 端 只 处 理 有 效 事件 中 的 一 部 分 。 

分 析 对 发 现 趋势 和 关键 指标 分 布 非常 有 用 。 你 可 以 用 它 来 发 现 每 个 用 户 的 平均 会 话 时 长 或 


平均 交易 量 。 但 不 要 将 其 用 于 跟踪 ,例如 ， 应 用 已 经 被 安装 了 多 少 次 。 一 些 特定 的 API 可 
以 准确 跟踪 这 些 统计 项 。 









































自 上 而 下 与 自 下 而 上 分 析 
自 上 而 下 的 分 析 系 统 位 于 客户 端 应 用 ,主要 监控 用 户 行为 。 受 监控 的 活动 包括 : AP 
随 着 内 容 流 跳 转 的 活动 、 用 户 正在 交互 的 视图 和 所 切换 的 视图 控制 器 ， 以 及 在 应 用 某 
个 模块 所 花费 的 时 间 等 。 
自 下 而 上 的 分 析 系 统 通 过 查看 相关 服务 问 的 信息 来 重建 用 户 行为 和 体验 。 例 如 ， 通 过 
服务 端 接 口 调用 来 发 现 用 户 如 何 请 求 内 容 流 的 条 H, 并 得 出 用 户 在 流 中 进展 的 程度 的 
有 关 结 论 。 
这 两 种 方法 都 有 自己 的 注意 事项 。 
自 上 而 下 的 方法 可 能 会 导致 网 络 流量 的 消耗 增多 ， 当 连接 断 开 或 网 络 不 可 用 时 会 丢失 
数据 。 如 果 使 用 本 地 持久 化 缓存 用 于 批量 处 理 ， 还 可 能 会 消耗 本 地 存储 空间 。 
自 下 而 上 的 方法 不 会 产生 真实 数据 ， 它 只 是 对 用 户 行 为 的 逐步 重建 。 你 可 能 无 法 得 知 
用 户 花费 了 多 长 时 间 浏 览 消息 流 的 一 条 内 容 ， 只 是 知道 用 户 请 求 了 哪 条 对 应 的 消息 。 











12.4 ”真实 用 户 监控 


真实 用 户 监控 是 监控 应 用 以 获取 和 分 析 用 户 的 每 个 事务 的 方法 。 它 依赖 于 服务 端 或 客户 端 
内 用 于 监控 的 服务 ， 这 些 服务 可 以 监控 活动 的 组 件 、 其 功能 、 应 用 的 响应 性 、 总 体 的 资源 
使 用 情况 和 其 他 各 种 参数 。 


它 还 有 各 种 别名 ， 包 括 终端 用 户 监控 、 真 实用 户 测量 、 真 实用 户 指标 ， 等 等 。 


12.41 分 析 与 真实 用 户 监 控 对 比 


分 析 同 样 提供 这 些 结果 。 或 许 你 在 想 真 实用 户 监控 到 底 有 什么 与 众 不 同 的 地 方 。 你 可 
说 ,，“ 它 们 就 是 对 应 用 进行 埋 点 ， 然 后 分 析 数 据 ”。 
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的 确 ， 分 析 和 真实 用 户 监 控 有 相同 的 用 途 ， 即 它们 都 可 以 对 应 用 进行 埋 点 、 分 析 数 据 ， 然 
后 产 出 报告 。 

分 析 和 真实 用 户 监控 的 巨大 差异 在 于 ， 分 析 只 使 用 一 部 分 数据 (样本 ) 进行 处 理 来 提供 高 
级 趋势 预测 。 

各 种 各 样 的 产品 提供 对 应 用 进行 埋 点 和 分 析 的 功能 ， 并 将 自己 列 为 分 析 工 具 而 不 是 真实 用 
户 监控 工具 ， 因 为 它们 只 记录 样本 。 


12.4.2 ”使 用 真实 用 户 监控 
因为 真实 用 户 监控 会 记录 所 有 的 事件 ， 而 不 仅 是 样本 ， 所 以 你 应 将 其 用 于 监控 关键 事件 。 
以 下 是 一 些 例子 。 


。 任何 错误 ， 包 括 应 用 崩溃 或 无 效 状态 。 
。 应 用 新 版 本 发 布 后 质量 的 变化 。 

。 新 功能 相关 的 用 户 行为 变化 。 

。 记录 重要 事务 中 的 每 个 步骤 。 


19.5 h2 
应 用 埋 点 与 功能 实现 同样 重要 。 它 是 深入 了 解 应 用 质量 、 应 用 健康 度 、 用 户 行为 的 重要 
途径 。 


使 用 分 析 工 具 筛 选 大 量 的 数据 ， 以 便 明 确 特征 模式 。 你 可 以 根据 应 用 创建 连续 动作 片断 来 
了 解 用 户 行为 ， 发 现 使 得 用 户 放弃 整个 动作 的 痛 点 或 步骤 。 这 在 引导 货币 交易 的 步骤 中 非 
常 有 用 ， 因 为 你 肯定 迫切 想 要 了 解 这 些 场景 下 交易 下 降 的 原因 。 


真实 用 户 监控 应 当 用 于 监控 任务 的 关键 步 又， 包括 但 不 限于 应 用 质量 和 用 户 行为 。 


再 次 强调 ， 不 要 阻塞 应 用 。 过 度 使 用 埋 点 不 仅 会 对 设备 上 的 客户 端 应 用 造成 严重 负担 ， 服 
务 端 程序 也 需要 对 上 报 数据 进行 处 理 。 


埋 点 不 能 替代 调试 。 在 开发 设备 上 应 当 只 使 用 调试 工具 。 
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第 五 部 分 
iOS 9 





iOS 9 于 2015 年 9 月 发 布 ， 引 入 了 许多 提高 应 用 性 能 的 改动 。 本 部 分 将 讨论 这 些 改动 ， 以 
及 如 何 有 效 地 使 用 它们 。 


BIS 





iOS 9 


iOS 9 包含 了 能 够 影响 应 用 运行 的 一 些 重要 改动 。 


除了 改进 标准 纪 





卓 件 以 及 操作 系统 的 性 能 外 ， 一 些 其 他 功能 也 可 以 提高 应 用 性 能 。 


本 章 将 讨论 以 下 内 容 。 
。 应 用 的 生命 周期 
iOS 引入 了 新 功能 ， 同 时 ， 影 响 性 能 的 一 些 原 有 功能 被 全 面 检修 。 


。 用 户 界 面 
新 加 入 的 视 


。 扩展 





图 已 可 用 ， 可 能 会 提高 浑 染 速度 。 全 新 的 网 页 展现 方式 也 已 就 绪 。 

















iOS 9 新 添加 了 两 个 扩展 。 如 有 果 尝 试 实现 它们 ， 你 需要 仔细 考虑 性 能 影响 。 





。 应 用 瘦身 











现在 不 仅 可 以 手动 优化 应 用 ， 你 还 可 以 使 用 苹果 的 优化 工具 ， 该 工具 根据 应 用 安装 的 设 
备 量 身 定做 一 一 期 待 已 久 的 功能 。 





AK iOS 9 的 完整 改动 可 参见 苹果 开发 者 网 站 (http://apple.co/1Bn94oT)。 


13.1 应 用 的 生命 周期 


iOS 9 提供 了 新 的 方式 来 启动 并 激活 一 个 应 用 ， 同 时 对 原 有 方式 进行 限制 。 





在 一 般 场景 下 ， 通 用 链接 取代 了 自 定 义 URL scheme。 后 者 仍然 是 可 用 的 ， 但 利用 该 功能 


检测 应 用 是 否 存在 被 严格 限制 了 。 通 过 这 种 方式 人 侵 用 户 的 隐私 ， 检 查 用 户 设备 中 上 百 个 








应 用 是 否 存在 上 


的 举动 不 再 可 行 。 
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全 新 的 Spotlight 应 用 内 搜索 提供 了 让 用 户 发 现 应 用 的 新 途径 ， 即 使 应 用 尚未 安装 在 设备 
中 。 这 意味 着 拥抱 新 功能 、 编 写 新 代码 、 让 其 顺利 运行 。 


13.1.1 通用 链接 

TE iOS 8 中 启动 其 他 应 用 ， 尤 其 从 Web 启动 ， 唯 一 的 方法 就 是 使 用 自 定义 的 URL scheme 
(我 们 在 5.2.3 节 中 的 “深层 链接 ”部 分 讨论 过 )。 这 项 技术 在 10S 9 上 仍然 可 用 ， 但 会 受到 
严格 限制 。 

应 用 仍 可 使 用 canOpenURL: 方法 来 检测 其 他 应 用 在 设备 上 的 可 用 性 。 但 是 ， 为 了 防止 该 方 
ECA, ‘iOS 9 限制 了 该 方法 的 调用 最 多 不 超过 50 个 唯一 的 自 定义 URL scheme, 

如 果 在 iOS 8 及 更 早 版 本 上 编译 应 用 ， 那 么 设备 将 记录 前 50 个 唯一 的 自 定义 
URL scheme, MÆ iOS 9 上 编译 则 必须 提供 完整 的 清单 ， 声 明 应 用 需要 打开 

的 自 定义 URL scheme (最 多 不 超过 50 个 )。 

































































iOS 9 引入 了 通用 链接 ， 人 允许 应 用 处 理 之 前 只 能 被 Safari 打开 的 http 或 https 链接 。 

通常 的 处 理 步 骤 如 下 。 

(1) 源 应 用 调用 openURL: 方法 打开 http 或 者 https 的 URL. 

(2) 操作 系统 检测 已 安装 的 应 用 是 否 可 以 处 理 该 URL, 

。 如 果 可 以 ， 则 启动 该 应 用 。 

。 否则 使 用 Safari 打开 。 

这 种 方法 的 优点 在 于 URL 总 能 被 打开 。 换 而 言 之 ，openURL: 方法 对 这 些 URL 总 是 返回 

YES， 你 不 必 担 心 需要 在 应 用 中 进行 分 支 处 理 。 

移动 网 页 可 以 使 用 智能 应 用 广告 条 ”引导 用 户 下 载 应 用 。 

要 想 使 用 这 项 功能 ， 你 需要 进行 如 下 设置 。 

(1) 为 应 用 添加 com. apple.developer.associated-domains 授权 ， 如 图 13-1 所 示 。 
域 的 取 值 必须 依照 applinks:{domain-to-handle} 格式 ， 子 域 不 可 以 使 用 通配符 。 每 个 
子 域 必须 进行 单独 注册 。 

(2) 创建 一 个 经 过 签名 且 名 为 apple-app-site-association 的 JSON 文件 ， 内 容 包 含 应 用 以 及 相 
关联 的 网 站 路 径 。 
使 用 与 应 用 签名 相同 的 密 钥 。 























注 1: 例如 ， 在 Twitter 上 线 应 用 图 谱 功 能 ， 用 于 收集 设备 上 安装 的 应 用 列表 后 ， 用 户 隐私 问题 备 受 关注 。 
注 2: iOS Developer Library, “Promoting Apps with Smart App Banners” (http://apple.co/1TzDzxp). 
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oO General Capabilities Resource Tags Info Build Settings Build Phases Build Rules 
PROJECT = 
B HighPerformance 


> € Keychain Sharin 
TARGETS A y a 


[| HighPerformanceTests d Gp Inter-App Audio 

E) actionRead 

E) shareRead » Qy Background Modes 
E) documentProvider 
E 


documentProvider... v m Associated Domains 


Domains: applinks:www.m10v.com 


+ 


Steps: ¥ Add the "Associated Domains" entitlement to your entitlements file 
vV Add the "Associated Domains" entitlement to your App ID 


> (96) App Groups 


"Oe: 

















dB 





EX jJ 
Data Protection (E 








图 13-1: 已 关联 的 域名 授权 
这 样 就 能 在 应 用 和 域名 间 建 立信 任 。 


T 











例如 ， 对 例 13-1 所 示 的 授权 applinks:ios.mydomain.com 及 其 关联 文件 ， 应 用 可 以 处 理 





http[s]://ios.mydomain.com/mypath/, http[s]://ios.mydomain.com/basepath/ 及 其 子路 径 。 


例 13-1 通用 链接 : 对 网 站 关联 文件 进行 签名 





{ 
"applinks": { @ 
"details": {@ 
"ABCDEFGHIJ.com.mydomain.bundleid": ( © 
"paths": [ @ 
"/mypath/", 
"/basepath/*" 
] 
} 
} 
} 
} 
@ applinks 服务 。 该 文件 同样 可 用 于 配置 Handoff 的 activitycontinuation 服务 。 
@ details 段 一 一 必须 以 此 命名 。 


Q9 应 用 完整 的 密 钥 值 ， 格 式 为 {team-id}. Capp-bundle-id), 
Q 路 径 列表 ， 可 使 用 通配符 。 


使 用 openssl 命令 行 工具 对 该 文件 进行 加 密 〈 见 例 13-2). 





iOS 9 
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f| 13-2 通用 链接 : 签名 应 用 网 站 关联 文件 
cat content.json | openssl smime 
-sign 
-inkey app-signer-private.key 
-signer app-signer-certificate.pem 
-noattr 
-nodetach 
-outform DER 
» apple-app-site-association 


为 了 处 理 这 个 URL, SCH UIApplicationDelegate 协议 的 application:continueUserActivi 
ty:restora tionHandler: 方法 。 例 13-3 展示 了 处 理 链接 的 一 个 示例 。 


例 13-3 通用 链接 : 处 理 链接 
- (BOOL)application:(UIApplication *)application 
continueUserActivity:(NSUserActivity *)userActivity 
restorationHandler:(void (^)(NSArray * restorableObjects))restorationHandler {@ 





NSURL *url = userActivity.webpageURL; @ 
// 处 理 URL 
return YES; 


} 


@ 应 用 委托 回调 。 
@ 使 用 webpageURL JE P'ESEZXH URL, 


13.1.2 BR 

应 用 搜索 提供 了 新 方式 以 搜索 某 个 应 用 内 的 公开 信息 , 即使 设备 上 尚未 安装 该 应 用 。 此 类 
信息 可 被 Handoff、Siri 和 Spotlight 检索 。 

如 果 已 经 安装 了 应 用 ， 那 么 可 以 通过 深层 链接 打开 。 如 果 未 安装 应 用 ，Safari 可 以 引导 用 
户 到 网 页 。 通 用 链接 确保 你 只 需要 提供 一 份 URL。 

任何 内 容 都 可 通过 苹果 公司 的 服务 器 索引 。 设 备 上 的 本 地 内 容 一 开始 只 能 在 本 地 搜索 使 
用 ， 当 同样 的 检索 发 生 在 多 个 设备 上 时 ， 内 容 将 被 发 送 到 苹果 公司 的 服务 器 ， 并 且 建 立 索 
引 以 供 更 多 设备 使 用 。 

应 用 搜索 可 以 帮助 用 户 发 现 未 安装 的 应 用 ， 并 在 应 用 安装 后 帮助 用 户 快速 访问 应 用 中 相关 
的 内 容 。 若 应 用 未 安装 ， 则 用 户 将 首先 进行 安装 。 安 装 完成 后 ， 用 户 可 以 直接 访问 相应 结 
果 ， 而 无 需 在 应 用 中 实现 自 定义 搜索 。( 此 外 ， 用 户 无 需 在 应 用 中 多 步 导 航 到 相关 页 面 ， 
这 样 就 提高 了 整体 的 用 户 体验 。) 

假设 你 将 使 用 以 上 的 一 个 或 多 个 特性 ， 那 么 你 需要 慎重 考虑 在 应 用 中 实现 时 的 性 能 。 

以 下 三 种 方式 可 以 将 内 容 提供 给 应 用 搜索 。 

。 NSUserActivity 类 中 的 新 方法 和 属性 可 用 于 检索 。 





































































































注 3: iOS Developer Library, “App Search Programming Guide” (http://apple.co/IRZA8RN). 
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* Core Spotlight HERH MARA ES nli e ERS, JEE.SCPE S AGE BEBE, 

。 Web 标记 让 网 页 内 容 可 被 检索 。 

1. NSUserActivity 

NSUserActivity 在 iOS 9 中 新 增 了 方法 ， 可 在 本 地 索引 中 添加 应 用 状态 。 以 下 的 新 属性 加 

强 了 本 地 索引 及 公开 索引 。 

。 keywords 属性 可 用 于 链接 关键 词 。 

e eligibleForSearch 属性 标识 数据 可 用 于 本 地 搜索 。 

e eligibleForPublicIndexing 属性 标识 数据 可 用 于 在 苹果 服务 器 上 的 公开 搜索 ， 因 此 可 
用 于 跨 设 备 检 索 。 


例 13-4 展示 了 有 此 作用 的 示例 代码 。 
例 13-4 添加 到 本 地 索引 : NSUserActivity 


NSUserActivity *activity = [[NSUserActivity alloc] 
initWithActivityType:Q"com.mydomain.plist-activity-type"]; @ 


























activity.title = Q"iOS 9 Features"; @ 
activity.keywords = [NSSet setWithObjects:@"ios 9", 

Q"new features", Q"wwdc 2015", nil]; © 
activity.userInfo = Q( ... 5; O 
activity.eligibleForSearch = YES; @ 
activity.eligibleForPublicIndexing = YES; © 
[activity becomeCurrent]; @ 


Q 根据 一 个 已 注册 的 活动 类 型 创建 一 个 活动 。 

O 设置 在 搜索 结果 中 展示 的 标题 。 

© 与 数据 关联 的 关键 词 。 

Q 设置 一 个 与 活动 相关 联 的 NSDictionary 类 型 数据 。 

Q 设置 活动 可 用 于 搜索 。 

Q 设置 活动 可 用 于 公开 索引 。 数 据 将 会 发 往 人 苹果 服务 器 并 建立 索引 。 它 将 会 在 公开 的 
Spotlight 与 Safari 搜索 结果 中 出 现 ， 并 且 根 据 人 气 指数 排名 。 

O 标识 应 用 当前 的 状态 (激活 该 活动 )。 一 量 完 成 ， 它 将 被 添加 到 通用 索引 (CssearchableIndex) 
中 。 

以 下 是 与 性 能 相关 的 一 些 提示 。 

。 提供 充足 的 关键 词 使 得 内 容 可 被 检索 ， 但 切 勿 滥用 。 注 意 ， 搜 索 可 能 是 本 地 运行 的 ， 多 
个 关键 词 不 仅 会 降低 排名 ， 同 时 也 会 降低 搜索 速度 。 

e userInfo 可 用 于 存储 与 活动 相关 的 自 定义 数据 。 因 为 数据 将 存储 在 应 用 外 ， 所 以 尽 可 能 
保持 小 的 体积 。 这 些 数据 在 建立 索引 时 会 被 序列 化 , 而 用 户 打 开 搜 索 结果 时 会 反 序列 化 。 
数据 越 多 ， 耗 时 越 长 。 

2. Core Spotlight 

iOS 9 引入 了 新 的 Core Spotlight" 框架 。 通 过 提供 对 应 用 内 容 的 索引 以 及 管理 设备 中 的 索 










































































注 4: iOS Developer Library, “Core Spotlight Framework” (http://apple.co/1 QFKyS5I). 
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引 ， 它 可 以 让 应 用 参与 到 搜索 中 来 。 这 些 内 容 可 以 由 应 用 提供 或 由 用 户 生成 。 


Core Spotlight 提供 了 一 个 API， 通 过 使 用 一 个 应 用 内 唯一 的 ID 让 内 容 可 被 检索 ， 同 时 可 
对 其 执行 更 新 或 删除 操作 。 一 旦 建立 了 索引 ， 数 据 就 可 以 通过 Spotlight 和 Safari 被 搜索 。 
如 果 这 项 数据 关联 了 一 个 NSUserActivity， 那 么 它 也 能 在 公开 的 索引 中 生效 。 


该 框架 通过 结构 化 数据 提供 更 多 的 控制 方式 。 特 别 是 ， 对 一 个 唯一 应 用 ID 的 要 求 可 以 使 
在 特定 应 用 中 强大 的 内 容 搜索 更 加 顺畅 。 


例 13-5 中 的 代码 展示 了 如 何 使 用 Core Spotlight 框架 















































例 13-5 使 用 Core Spotlight 
// 添 加 内 容 到 索引 
#import «CoreSpotlight/CoreSpotlight.h» @ 
#import <MobileCoreServices/MobileCoreServices.h> @ 


-(void)addToIndex:(...) ( © 
CSSearchableItemAttributeSet *attrs = [[CSSearchableItemAttributeSet alloc] @ 
initWithItemContentType:(NSString *)kUTTypeText]; @ 


attrs.title = @"Mango"; 
attrs.contentDescription = @"King of Fruits"; 
attrs.keywords = @["mango", "fruit", "vegetation"]; © 


CSSearchableItem *item = [[CSSearchableItem alloc] 
initWithUniqueIdentifier:@"mango" 
domainIdentifier :@"com.mydomain.item-domain" 
attributeSet:attrs]; @ 


[[CSSearchableIndex defaultSearchableIndex] @ 
indexSearchableItems:Q[item] © 
completionHandler:^(NSError *e) { @ 

if(e) { 
// 错 误 处 理 
) else ( 
// 一 切 正 常 





11; 
} 


// 在 AppDelegate 中 处 理 搜索 结果 链接 

-(BOOL)application:(UIApplication *)application 
continueUserActivity:(NSUserActivity *)userActivity 
restorationHandler:(void (^)(NSArray *))restorationHandler { @ 








if([CSSearchableItemActionType isEqualToString:userActivity.activityType]) ( @ 
NSDictionary *details - userActivity.userInfo; (9 
NSString *itemId - [details 
objectForKey:CSSearchableItemActivityIdentifier]; (D 


// 使 用 唯一 ID 处 理 

















注 5: 5 NSUserActivity 相 比 ， 这 里 没有 要 求 唯一 的 ID。 开 发 者 也 许 会 将 其 放 入 userInfo 字典 ， 不 过 这 个 
操作 是 可 选 的 








return YES; 
} 


使 用 Core Spotlight 接口 需要 导入 CoreSpotlight h@ 头 文件 。 
使 用 UTI 类 型 常量 需要 导入 MobileCoreServices.h) 头 文件 。 


本 例 中 使 用 辅助 方法 addToIndex® 将 内 容 添 加 到 索引 。 为 简便 起 见 ， 在 此 省 略 该 方法 的 
参数 。 

CSSearchableItemAttributeSet 类 @ 可 用 来 定义 与 索引 内 容 相 关 的 属性 。 比 如 ，title、 
contentDescription 这 两 个 重要 的 属性 定义 搜索 结果 的 输出 。 图 13-2 展示 了 示例 中 Spotlight 
关于 内 容 的 搜索 结果 。 根 据 条 目的 类 型 ， 你 可 以 设置 一 个 或 多 个 其 他 可 用 的 属性 。” 

在 本 例 中 ， 该 项 内 容 类 型 设置 为 纯 文本 回 . ' 

添加 title 和 可 选 的 contentDescription® 非常 重要 ， 它 们 控制 了 搜索 结果 的 展示 效果 
( 见 图 13-2), 












































Search Web Search 


id=mango 
Search App Store title=Mango 


Search Maps OK 


a w E a a y d als 
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13 © space Search 














13-2: Spotlight 搜索 (Zr) 与 应 用 展示 的 搜索 结果 ( 右 ， 演 示 用 的 警告 弹 框 ) 


一 旦 配置 好 属性 集合 ， 就 可 以 使 用 它们 来 让 最 终 的 搜索 项 CSSearchableItem 对 象 可 被 索引 @。 
创建 时 需要 一 个 用 于 应 用 的 uniqueIdentifier。domainIdentifier 的 确切 目的 在 编写 时 还 




















注 6: 参见 苹果 开发 者 网 站 的 完整 列表 (http://apple.co/1SvhU5y)。 

注 7: 我 们 在 8.3 市 中 讨论 过 文档 类 型 。 

注 8: 它 可 以 仅仅 是 元 数据 ， 也 可 以 对 应 用 中 的 一 项 做 进一步 分 类 ， 以 便 唯一 标识 符 基本 上 是 对 单独 应 用 的 
单独 领域 生效 。 这 将 允许 多 个 SDK/ 组 件 在 一 个 应 用 中 无 颖 工作 。 
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CSSearchableIndex(9 是 帮助 名 
WRI | o 


indexSearchableItems:completionHandler: 方法 允许 异步 检索 多 个 项 @@。 
在 完成 处 理 回调 @ 中 可 绪 得 索引 结果 ， 完 成 处 理 回调 不 能 为 nil, 


另 一 方面 ， 当 用 户 点 击 来 自 应 用 的 搜索 结果 时 ， 应 用 委托 方法 application:continueUserA 
ctivity:restorationHandler:@ 将 被 调用 。 


= 


建 索 引 的 主要 的 类 。defaultSearchableIndex 是 设备 上 的 全 





























activityType 在 此 示例 中 总 是 CSSearchableItemActionType。 


QuserInfo® 字典 至 少 需要 存储 一 项 。 该 项 必须 使 用 CSSearchableItemActivityIdentifier(D 

的 键 来 提供 之 前 创建 CSSearchableItem 使 用 的 唯一 ID。 使 用 该 值 来 查找 应 用 中 对 应 的 资 

料 ， 并 展示 细节 。 

当 使 用 Core Spotlight 接口 时 ， 你 应 当 遵循 以 下 与 性 能 相关 的 建议 。 

e 在 title 和 description 中 提供 充足 的 信息 ， 但 不 要 使 它们 过 于 匈 长 ， 否 则 不 但 用 户 无 

法 看 全 数据 ， 而 且 会 影响 序列 化 / 反 序列 化 的 耗 时 。 

不 要 过 度 使 用 keyword， 除 了 可 能 招致 惩罚 ， 还 会 影响 索引 创建 和 搜索 性 能 。 

。 尽 可 能 减少 用 户 活动 的 userInfo 字典 中 提供 的 内 容 。 只 提供 快速 展示 特定 结果 所 需 的 
数据 。 

3. Web 标 记 

AK iOS 6 起 ,如 果 用 户 在 iOS 设备 Safari 移动 浏览 器 中 访问 应 用 网 页 ， 智 能 应 用 广告 条 "已 

经 成 为 提高 应 用 下 载 量 的 标志 方法 。 

芋 果 公司 已 经 引入 了 新 的 增强 功能 。Applebot (https://support.apple.com/en-us/HT204683 ) 

将 会 检索 网 页 寻找 元 标签 ， 以 便 为 Spotlight 和 Siri 搜索 使 用 的 公共 索引 提供 数据 。 在 

2015 年 的 全 球 开发 者 大 会 上 ， 蕴 果 公 司 提 到 它 将 支持 标准 元 标签 ，"” 其 中 包括 但 不 限于 在 

Schema.org 上 定义 的 ， 以 及 遵循 Open Graph 协议 的 标签 。 在 这 些 公 开标 准 还 不 能 满足 需 

求 的 特定 场景 下 ， 苹 果 公 司 已 经 配置 了 新 的 标签 。 


对 于 在 搜索 中 展现 的 应 用 内 容 ， 你 可 以 使 用 这 些 标签 来 提供 特定 的 页 面 展 示 。 


13.1.3 ”搜索 最 佳 实践 

虽然 通用 链接 和 搜索 看 起 来 很 不 一 样 ， 但 二 者 其 实 是 相辅相成 的 。NSUserActivity 和 Core 
Spotlight 使 用 本 地 内 容 建立 索引 ， 并 且 也 可 以 向 苹果 服务 器 提供 。 当 网 页 标记 能 够 链接 到 
应 用 时 ， 它 使 用 Applebot 帮助 苹果 服务 器 建立 索引 。 

但 是 ， 这 些 结果 只 会 在 满足 一 定 人 气度 和 其 他 因素 时 出 现 。 绝 大 多 数 的 其 他 因素 目前 还 不 
明确 ， 但 是 开发 者 必须 提供 充分 的 信息 来 充实 索引 。 这 将 确保 用 户 可 以 发 现 你 的 应 用 和 其 
中 的 内 容 ， 并 与 其 交互 。 













































































































































































注 9: iOS Developer Library, “Promoting Apps with Smart App Banners” (http://apple.co/INnawJc). 
iE 10: iOS Developer Library, “Mark Up Web Content” (http://apple.co/1qMnZ7L). 
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以 下 列表 包含 了 我 的 一 些 初 步 建议 。( 注 意 ， 此 列表 中 的 最 佳 实践 并 未 经 过 实战 检验 。 它 
是 不 全 面 的 ， 随 着 我 们 进一步 采用 新 技术 和 研究 理 点 数据 ， 它 会 不 断 改 进 。) 




















使 用 通用 链接 。 网 页 和 应 用 应 当 对 同一 个 链接 展示 相同 的 内 容 。 

对 同一 内 容 建立 索引 时 使 用 同一 ID， 这 将 确保 对 相同 内 容 获 得 更 好 的 访问 排名 。 
使 用 description 属性 提高 用 户 体验 。 

尽 可 能 提供 缩 略 图 。 
明智 地 选择 关键 词 。 

目前 还 没有 资料 表明 使 用 多 少 个 关键 词 会 导致 排名 惩罚 ,但 有 充分 的 理由 推测 对 关键 词 
的 滥用 会 招致 惩罚 。 

以 有 限 状态 机 的 方式 来 实现 应 用 ， 确 保 引 导 到 应 用 的 搜索 结果 可 以 被 优雅 地 处 理 。 

用 户 也 许 会 多 次 通过 相同 /不 同 的 项 或 通用 链接 打开 应 用 。 

如 图 13-2 所 示 ， 右 侧 的 截图 表明 ， 处 理 结果 的 应 用 有 一 个 返回 搜索 的 按钮 ， 以 便 用 户 
可 以 返回 Spotlight。 但 是 ， 此 时 可 获取 的 通知 只 有 application WillResignActive 以 及 
后 续 的 事件 。 
严格 来 说 ， 这 意味 着 ， 你 不 清楚 用 户 是 否 接 到 了 来 电 、 点 击 了 Home 按钮 ， 还 是 按 下 了 
返回 搜索 按钮 。 


这 也 意味 着 ， 一 旦 应 用 进入 后 台 ， 你 并 不 知道 如 果 处 理 上 一 次 的 搜索 结果 或 在 应 用 中 打 
开 的 通用 链接 一 一 是 应 当 保留 打开 的 状态 ， 还 是 将 应 用 返回 之 前 打开 链接 前 的 状态 呢 ? 


系统 提供 的 返回 按钮 让 用 户 回 到 源 应 用 。 应 用 获得 applicationDidBecomeActive: 的 回 
调 ， 但 是 区 分 不 了 是 用 户 进行 应 用 的 切换 还 是 点 击 了 返回 按钮 。 因 此 ， 建 议 在 应 用 内 提 
供 一 个 自 定 义 的 返回 按钮 ， 这 样 一 来 ， 当 用 户 返 回应 用 时 ， 上 一 次 的 结果 仍然 可 以 显示 
出 来 ， 同 时 也 提供 了 一 个 回 到 之 前 状态 的 选项 。 


下 一 个 要 处 理 的 复杂 问题 是 ， 如 何 管理 链接 到 应 用 的 多 个 结果 / 链接 。 在 用 户 进行 多 次 
返回 操作 的 过 程 中 ， 应 用 也 许 就 已 经 结束 而 看 不 到 原来 的 状态 。 


是 否 在 回 退 栈 中 支持 多 项 调用 取决 于 产品 决策 。 
当 多 个 项 存在 于 回 退 栈 时 ， 是 否 提供 一 个 直接 链接 到 达 应 用 原先 状态 同样 取决 于 产品 决策 。 
工程 师 将 必须 支持 这 项 功能 。 



















































































































































































13.2 用 户 界 面 











iOS 9 有 大 量 关于 用 户 界面 层 的 更 新 。 无 论 你 是 否 使 用 它们 都 将 会 影响 你 的 应 用 的 性 能 。 











为 了 讨论 方便 ， 我 们 将 它们 分 为 两 大 类 : 


UIKit 框架 的 改动 
Safari 服务 框架 的 改动 





游戏 相关 的 框架 ， 如 GameplayKit, Model I/O, MetalKit, Metal, SceneKit 和 SpriteKit 框 
架 都 不 在 讨论 范围 。 
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13.2.1 UIKit 框 架 
iOS 9 提供 了 一 个 新 的 容器 视 





a 














常见 )，UIStackView 将 会 为 你 提 


回想 之 前 讨论 过 的 邮件 视图 








市 中 的 图 
使 用 


6-14)。 








( 见 6.2.6 节 中 的 
味 着 多 个 约束 需要 小 心 的 手工 处 理 ， 


自动 布局 的 开销 不 仅 体现 在 
工 设置 它们 ， 让 其 在 iPhone 和 iPad 的 不 同 屏幕 上 生效 绝 非 易 


一 一 UIStackView (http:/apple.co/1NoqljW )， 用 于 
者 在 水 平 或 垂直 方向 演 染 多 个 视 


当 你 需要 创建 一 个 表单 风格 的 界 


图 。 








供 一 个 更 快捷 、 更 方便 的 途径 。 





EU 


运行 





以 及 解 线性 方程 组 时 的 


= ps 


运行 


时 ， 也 体现 在 设计 时 。 





= 





o 


面 ， 在 水 平 或 垂直 方向 排列 多 个 视 











帮助 开发 





图 时 (这 种 情况 似乎 更 





图 6-11) ， 它 有 七 个 子 视图 。 使 用 自动 布局 意 
时 的 额外 开销 (参见 6.3 


图 6-11 有 超过 20 个 约束 。 手 





UIStackView 是 一 项 广 受 欢迎 、 


便 ， 最 终 惠 及 终端 用 户 。 


HAR ERE ILM DH, 





借鉴 安 卓 
让 人 址 首 以 盼 的 功能 。 


安 车 在 早期 就 有 了 LinearLayout (http://developer.android.com/reference/android/widget/ 
LinearLayout.html), ， 很 高 兴 革 果 团 队 能 向 其 他 团队 学 习 ， 从 而 为 应 用 开发 人 员 提 供 方 


在 雅虎 公司 ， 我 们 实现 了 自己 版 本 的 线性 布局 ， 并 获 益 良 多 。 对 于 拥有 10-12 个 UI 
元 素 的 视图 ,运行 时 创建 和 布局 时 间 减 少 了 19%。 对 UITableView 中 循环 使 用 的 视图 ， 
重新 布局 也 减少 了 超过 10% 的 时 间 。 


在 开发 阶段 ， 一 个 复杂 的 视图 设计 将 花费 三 天 来 满足 跨 iPhone 和 iPad 的 使 用 ， 但 使 用 











Xcode 7 的 故事 板 编辑 器 支持 将 多 个 视 





栏 底部 偏 右 的 位 置 看 到 一 个 新 
选中 的 视图 都 将 被 推 入 一 个 新 的 








a 











HEA UIStackView。 如 图 





UIStackView, 


13-3 所 示 ， 你 可 以 在 工具 
图 标 。 选 中 多 个 视图 ， 然 后 点 击 这 个 人 栈 形状 的 按钮 。 所 有 
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Alignments 
EFE 


Stack View 


Select 














Distributiona é Select 
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EFE 
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13-3: 故事 板 一 一 在 UIStackView 中 排列 多 个 视图 
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以 下 属性 控制 了 UIStackvtew 中 受 管 理 视图 的 这 染 效 果 。 


* axis 

















定义 了 布局 的 方向 ， 可 以 是 UILayoutConstraintAxisHorizontal 或 UILayoutConstrain 


tAxisVertical, 


UILayoutConstraintAxisVertical 表示 受 管理 的 视图 将 会 沿 垂直 方向 进行 演 染 。 后 添加 
的 视图 将 会 在 前 一 个 演 染 好 的 视图 下 方 。UILayoutConstraintAxisHorizontal 表示 受 管 
理 的 视图 将 会 沿 水 平方 向 进行 演 染 。 后 添加 的 视图 将 会 在 前 一 个 演 染 好 的 视图 右 侧 。 

















默认 值 是 UILayoutConstraintAxisVertical。 











e alignment 








控制 视图 的 对 齐 方式 。 与 UIStackView 垂直 坐标 轴 对 齐 ， 默 认 值 是 UIStackvtewAlignmentFill, 





e distribution 





控制 视图 的 大 小 。 该 值 影响 的 大 小 和 位 置 沿 着 UIStackview 的 坐标 轴 分 布 。 默 认 值 是 


UIStackViewDistributionFill, 
e spacing 

以 pt 为 单位 声明 了 相 邻 两 个 视图 的 间隔 距离 ， 默 认 值 为 0。 
e baselineRelativeArrangement 

控制 了 受 管理 视图 间 的 垂直 间距 是 否 根据 基线 测量 得 到 。 如 果 是 Yes, 
个 文字 视图 的 文字 最 后 一 行 到 甚 后面 视图 的 文字 第 一 行 计算 。 


e layoutMarginsReLativeArrangement 







































































那么 间距 是 从 一 


决定 了 UlStackView 平 铺 受 管理 的 视图 时 是 否 参 照 了 它 的 布局 边 距 或 边界 。 默 认 值 是 





No， 表 示 使 用 边界 。 


图 13-4 展示 了 这 些 属 性 在 横 轴 上 的 关联 性 。 对 于 纵 轴 而 言 ， 只 须 将 这 些 所 
90 FE. 











性 的 关联 轴 旋 转 





Spacing 
= 
z 
Distribution 











13-4; UIStackView 一 一 属性 及 其 关联 性 ( 横 轴 ) 
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UIStackView 是 UIView (Yak i RATA, Ls UIView 的 其 他 子 类 不 同 ， 它 不 
在 画布 上 进行 泻 染 。 因 此 ， 如 果 重 写 drawRect: 方法 ， 它 对 最 终 效果 并 没有 
任何 影响 。 改 变 backgroundColor 等 其 他 属性 也 是 一 样 的 结果 。 

UIStackView 使 用 arrangedSubviewts 的 概念 来 布局 ， 使 用 +addArrangedSubview 
而 不 是 addSubview 来 添加 视图 到 布局 当中 。 

或 者 ， 你 可 以 使 用 insertArrangedSubview:atIndex: 在 非 未 尾 位 置 插入 视图 。 
在 UIStackView 中 使 用 removeArrangedSubview: 删除 一 个 视图 ， 同 时 在 子 
视图 被 移 除 时 调用 removeFromSuperview。 


















































13-5 展示 了 XCode 编辑 器 中 UIStackView 对 应 的 属性 。 

到 目前 为 止 ，UIStackView 相关 的 最 佳 实践 如 下 。 

。 尽 可 能 使 用 它 ， 这 样 不 仅 能 减少 设计 视图 的 时 间 ， 还 能 提高 运行 时 的 性 能 。 
。 使 用 上 述 属 性 控制 最 终 的 布局 。 
。 如 果 对 它 的 性 能 不 满意 ， 可 能 是 因为 视图 过 于 复杂 了 ， 那 么 就 使 用 自 定 义 布局 。 














Stack View 


Axis Vertical 
Alignment Leading 
Distribution Fill 
Spacing 0° 


Baseline Relative 
~ | Layout Margins Relative 











13-5 UIStackView 一 一 Xcode 属性 编辑 器 


13.2.2 ”Safari 服务 框架 


iOS 7 中 增加 了 Safari 服务 框架 ， 该 框架 不 为 人 知 是 因为 其 提供 的 接口 主要 是 为 了 添加 URL 

BFA PY Safari 阅读 列表 。 本 节 将 讨论 在 iOS 9 中 新 增 的 SFSafariviewControLLer。 在 过 

去 ， 应 用 大 多 会 使 用 UIWebView RSM MAA (我 们 在 6.2.5 节 中 讨论 过 )。iOS 8 引入 的 

WebKit 框架 提供 了 更 高 性 能 的 WKWebView 类 。 有 些 应 用 便 从 UIWebView 迁移 到 WKWebView, 

虽然 性 能 有 所 提高 ， 但 有 些 问题 仍然 没有 解决 。 

。 渲染 引擎 总 是 基于 一 个 非 最 新 版 本 的 WebKit (https://www.webkit.org)， 至 少 落后 于 
设备 上 的 Safari 浏览 器 好 几 个 版 本 。 这 意味 着 在 Safari 中 浏览 网 页 内 容 总 是 优 于 使 用 
UIWebView 和 WKWebView 内 置 浏览 器 。 

。 没有 共享 cookie 的 方法 。 因 此 ， 如 果 用 户 已 经 使 用 浏览 器 (Safari, Chrome 或 者 其 他 ) 
登录 了 一 个 网 站 ， 在 使 用 内 置 浏览 器 时 仍然 需要 重新 登录 。 这 就 只 有 两 个 选项 :要么 让 
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用 户 离开 应 用 到 Safari 或 Chrome， 要 么 让 用 户 重新 登录 。 图 13-6 展示 了 HipChat 应 用 
的 设置 选项 。 





《 Settings 





OPEN LINKS IN 


HipChat 
Safari v 


Chrome 











13-6: HipChat 打开 链接 的 选项 





SFSafariViewController 党 试 解 决 这 些 问题 。 注 意 ， 它 不 是 一 个 视图 而 是 一 个 视图 控制 器 。 
这 意味 着 你 不 能 控制 UL 控件， 如 地 址 栏 和 屏幕 下 方 围 绕 HTML 内 容 的 动作 按钮 ， 如 图 
13-7 所 示 。 

视图 控制 器 使 用 Safari 浏览 器 的 cookie， 并 在 单独 的 进程 中 运行 。 这 意味 着 ， 如 果 用 户 使 
用 Safari 登录 过 ， 浏 览 操作 就 能 无 颖 进行 。 

应 用 的 cookie 并 不 共享 给 Safari 视图 控制 器 ， 这 意味 着 ， 如 果 用 户 仅 在 应 用 

中 登录 过 ， 那 么 还 需要 再 次 登录 。 

一 旦 用 户 登 录 ， 无 论 是 使 用 视图 控制 器 还 是 Safari 应 用 ， 会 话 都 将 继续 下 去 。 
































例 13-6 展示 了 使 用 SFSafariviewController 的 示例 代码 。 


例 13-6 {EJH SFSafariViewController 
-(void)showURL:(NSURL *)url ( 


SFSafariViewController *safari = [[SFSafariViewController alloc] 
initWithURL:url entersReaderIfAvailable:NO]; 


safari.delegate - self; 
[self presentViewController:safari animated:YES completion:nil]; 


} 


-(void)safariViewControllerDidFinish:(SFSafariViewController *)controller { 


// 用 户 点 击 完成 按钮 


[controller dismissViewControllerAnimated:YES completion:nil]; 


} 
13-7 展示 了 Safari 视图 控制 器 的 演示 效果 。 
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Carrier > 10:27 PM = 
m10v.com © Done 
Master’s Castle 
! Mindless Mind € - Please don't mind 


ing Mojito, Part 2: Common affinity controller 





In Part 1, we got our basic application 

ready with the following setup: 

e Application and Routing 
Configuration 

* TodoMVC defined assets 

© Initial UI - HTML markup and CSS stylesheet 

Tn Part 2, we explore configuring all of the CRUD actions 

and update all associated artifacts — model, controller and 

binder. Specifically, we explore the following actions — one 

ata time: 

Action — List all TODOs 

Action — Add a TODO 

Action — Edit a TODO 

Action — Mark one or all complete 

Action - Delete a TODO 


Step 1: Action - List all TODOS 


The first thing that we build up is retrieving all TODO 
items and showing them up. If there are no items in the 
list, then, as per the specs, the #main element must be 
hidden. 


Let us start with updating the model followed by the 
controller and binder in that order. 


arerp 


7 Read more ... 


000000 


Categories: Mojito | Tags: JavaScript. Mojito. MVC. TodoMVC. Web 





























f 








B 13-7: Safari 视图 控制 器 


TERR, Safari 浏览 器 


所 示 )。 


J 会话 不 会 被 带 到 SFSafariViewController 





图 控制 器 中 ( 见 图 13-8 





视 





Carrier S 























Filter by Category 





11:31 AM 
& yahoo.com [vi 


Carrier 




















Yahoo Sites ^B 
WA Mal " 

& ke 
E News 


ATA uade 





all 


jp ati 











& yahoo.com 





11:30 AM - 


G Done 








图 13-8. 即便 用 户 在 Safari 应 用 中 登录 了 (Ip), ， 他 在 使 用 SFSafariViewController 的 应 用 中 并 不 会 


自动 登录 ( 右 ) 





334 | 第 13 章 





我 推荐 使 用 SFSafariViewController, (Ha, ALA EAE IOS 9 中 可 用 ， 所 以 你 仍然 需要 
Xt iOS 7 的 用 户 提供 UIWebView, Xf iOS 8 的 用 户 提供 WKWebView, 


使 用 视图 控制 器 时 不 要 启用 entersReaderIfAvailable。 让 用 户 来 决定 是 否 进入 阅读 模式 。 


13.3 扩展 
iOS 9 引入 了 两 个 新 的 扩展 项 ( 见 图 13-9)， 这 两 个 扩展 项 会 影响 用 户 与 应 用 的 交互 以 及 对 
用 户 展示 的 UI。 
。 内 容 拦截 扩展 
在 使 用 Safari 或 SFSafariViewController 时 人 允许 限制 内 容 展示 。 


。 Spotlight 索引 扩展 
即便 应 用 不 运行 时 也 人 允许 更 新 设备 上 应 用 搜索 中 的 索引 。 

















Choose atemplate for your new target: 





| iOS A 
Application sO M, 3 r 
Framework & Library 
EAA ETR Action Content Blocker Custom Document 

PR Extension Extension Keyboard Provider 

Apple Watch 

w ® © © a 
watchOS . 

Application Photo Editing Share Extension Shared Links Spotlight Index 

Framework & Library Extension Extension Extension 
OS X 1 17 

Application = 

Framework & Library Today 

Application Extension Extension 








Test 
System Plug-in 
Other 














Cancel 














B 13-9: iOS 9 扩展 





根据 不 同 的 应 用 ， 你 可 能 需要 实现 一 个 或 多 个 上 述 扩展 。 

例如 ， 你 可 能 会 创建 一 个 需要 保护 儿童 隐私 且 限 制 浏览 成 人 内 容 的 应 用 。 内 容 拦 截 扩展 就 
是 在 设备 的 Safari 浏览 器 中 实现 如 此 插件 的 一 个 可 行 方案 。 

同 理 ， 当 不 再 使 用 时 允许 更 新 被 索引 的 内 容 也 是 一 个 好 主意 。 这 样 可 以 保证 Spotlight 搜索 
中 不 会 出 现 过 期 的 结果 。 而 Spotlight 索引 扩展 将 帮助 你 完成 这 些 。 
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13.3.1 ABER RE 


不 知 你 是 否 在 桌面 电脑 上 使 用 过 Adblock Plus (https://adblockplus.org) 来 过 滤 烦 人 的 广 
E? 如 果 用 过 ， 那 么 你 很 可 能 希望 移动 设备 也 有 此 功能 。 内 容 拦 截 扩 展 可 以 帮助 实现 这 个 
功能 。 

该 扩展 与 Safari 集成 (http://apple.co/1SvqjpG) 并 允许 过 滤 浏 览 网 页 的 内 容 。 


如 果 在 系统 设置 中 打开 Safari， 你 将 会 发 现 一 个 新 的 内 容 拦 截 入 口 。 你 可 以 在 设备 上 可 以 
安装 一 个 或 多 个 内 容 拦截 器 并 有 选择 地 局 用 它们 。 

内 容 拦 截 扩 展 使 用 一 个 JSON 格式 的 配置 文件 ， 从 而 控制 哪些 元 素 可 见 ， 哪 些 需 要 过 滤 。 
JSON 内 容 的 规范 可 在 WebKit 网 站 上 查看 (https://webkit.org/blog/3476/content-blockers- 
first-look/) 。 

例 13-7 展示 了 内 容 拦截 扩展 的 一 个 示例 。 它 要 求 一 个 类 来 实现 NSExtensionRequestHandling 
协议 。 该 协议 仅 有 一 个 方法 beginRequestWithExtensionContext:， 它 调用 -[NSExtensionContext 
completeRequestReturningItems:completionHandler:] 方法 ,传人 定义 好 的 过 滤器 


例 13-7 PATE E 
// 代 码 


- (void)beginRequestWithExtensionContext:(NSExtensionContext *)context { 
























































NSURL *url = [[NSBundle mainBundle] 
URLForResource:Q"blockerList" withExtension:Q"json"]; @ 
NSItemProvider *attachment - [[NSItemProvider alloc] 
initWithContentsOfURL:url]; @ 


NSExtensionItem *item - [[NSExtensionItem alloc] init]; 
item.attachments = Q[attachment]; © 


[context completeRequestReturningItems:Q[item] completionHandler:nil]; @ 


j 


Q 过 滤器 定义 (ISON 内 容 ) 的 URL, 
@ 用 initWIthContentsOfURL 初始 化 器 创建 一 个 NSItemProvider 实例 。 
© 创建 一 NSItemProvider 附件 的 NSExtensionItem 对 象 。 











@ 最 后 ， 通 过 调用 completeRequestReturningItems WENA, BULL Pil 13-8 了 解 过 滤器 定 
Mir. 
例 13-8 ”过 滤器 定义 示例 
[t 
"trigger": { 
"url-filter": "webkit.org/images/icon-gold.png" @ 
Ts 
"action": { 
"type": "block" @ 
} 


1] 
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Q 此 示例 基于 将 处 理 的 URL 决定 过 滤器 何 时 触发 。 
O 当 触 发 器 条 件 满足 时 应 该 采取 的 动作 是 拦截 内 容 ， 不 展示 出 来 。 


图 13-10 展示 了 如 何 配 置 内 容 拦 截 器 。 











Carrier > 10:47 PM = Carrier > 10:47 PM = 
€ Settings Safari € Safari Content Blockers 


PRIVACY & SECURITY 


ALLOW THESE CONTENT BLOCKERS: 
Do Not Track a ES: rac (UNO itr ot 


i | . HPerf Apps D 
Block Cookies Allow from Websites | Visit 


Content blockers affect what content is loaded 
while using Safari. They cannot send any 
information about what was blocked back to the app. 











Fraudulent Website Warning ( ) 


About Safari & Privacy... 


Clear History and Website Data 


READING LIST 
Use Cellular Data ` 


Use cellular network to save Reading List items 
from iCloud for offline reading. 





Content Blockers 





Advanced 











13-10; Safari 内 容 拦 截 器 设置 














图 13-11 展示 了 内 容 拦 截 器 如 何 影响 一 个 给 定 网 页 的 显示 效果 ， 图 中 所 用 网 页 是 http:// 
www.webkitorg。 左 侧 截图 是 内 容 拦 截 器 打开 时 的 效果 ， 右 侧 是 关闭 时 的 效果 。 注 意 ， 当 
内 容 拦截 器 打开 时 ，WebKit 应 用 的 标识 不 会 展示 出 来 。 
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Carrier > 


= 
Reader View Available © 


The WebKit Open Source Project 


Welcome to the website for the WebKit Open 
Source Project! 

WebKit is an open source web browser engine. WebKit is 
also the name of the OS X system framework version of the 
engine that's used by Safari, Dashboard, Mail, and many 
other OS X applications. WebKit's HTML and JavaScript code 
began as a branch of the KHTML and KJS libraries from KDE. 


Getting involved 
There are many ways to get involved. 
You can: 





© download the latest nightly build 
e install developer tools and then check out and build the 
Source code 
Once you have either of these, you can help by: 
© reporting bugs you find in the software 


Carrier > 





Working with the Code. 
Tools 


Reader View Available [v] 


The WebKit Open Source Project 


Welcome to the website for the WebKit Open 
Source Project! 

WebKit is an open source web browser engine. WebKit is 
also the name of the OS X system framework version of the 
engine that's used by Safari, Dashboard, Mail, and many 
other OS X applications. WebKit's HTML and JavaScript code 
began as a branch of the KHTML and KJS libraries from KDE. 


Getting involved 
There are many ways to get involved 
You can: 





© download the latest nightly build 
e install developer tools and then check out and build the 
Source code 
Once you have either of these, you can help by: 
© reporting bugs you find in the software 


© providing reductions to bugs 
e submitting patches for review 


More info 





contact us. 


Projects 


contribute to: 
e help us improve Website compatibility 
e write documentation 
e SVG 
e MathML 
a CSS 
e DOM 





Cie Bu 





More information about Webkit can be found on its wiki. You 
can help here too, by adding information that can help 
others learn about Webkit. If you have more questions, 


There are many exciting (new) projects that you can 





-j 











© providing reductions to bugs 
e submitting patches for review 


More info 

More information about Webkit can be found on its wiki. You 
can help here too, by adding information that can help 
others learn about WebKit. If you have more questions, 
contact us. 





Projects 
There are many exciting (new) projects that you can 
contribute to: 

e help us improve Website compatibility 

© write documentation 

e SVG 

e MathML 

e CSS 

e DOM 








URN) 

















B 13-11; 


因为 移动 端 Safari 4 
建议 。 








。 每 次 启用 时 尽 可 能 使 用 本 地 文件 系 
(例如 , Adblock Plus 可 能 会 从 服务 器 实现 与 更 新 ), 使 用 后 台 


。 对 于 动态 的 过 滤器 
以 便 与 服务 器 周期 性 地 同步 。 
。 最 小 化 过 滤器 





第 一 次 支持 扩展 ， 所 以 真实 的 最 佳 实践 还 设 有 出 现 。 以 下 是 一 





统 的 文件 ， 


条 目的 数量 ， 这 在 对 复杂 网 页 进 


在 WebKit 网 站 开启 内 容 拦截 器 的 效果 (A) 与 关闭 的 效果 (6) 


些 基 本 的 


而 不 是 从 网 络 上 同步 。 
下 载 ， 


EITEN S TEE. 


s 不 要 滥用 过 滤 。 EURENA ETIE, 并 且 让 应 用 可 以 访问 〈 提 示 : 使 用 


openURL: 方法 )。 


13.3.2 ”Spotlight 索 引 扩展 


一 般 来 说 , 你 会 
创建 一 个 维护 索引 的 扩展 ， 








注 11: 


在 应 用 打开 时 更 新 Spotlight 搜索 的 索引 。 


通过 Spotlight 索引 扩展 ， 你 可 以 





让 操作 系统 调度 不 在 运行 状态 的 应 用 ， 


Sebastian Noack, “ Adblock Plus and (a Little) More" 


给 予 其 一 个 更 新 索引 及 





(https://adblockplus.org/blog/content-blocking-in-safari- 


9-and-ios-9-good-news-or-the-death-knell-of-ad-blocking-on-safari). 





验证 索引 项 有 效 性 的 机 会 〈 例 如 ， 验 证 该 项 是 否 可 用 且 未 过 期 )。 
使 用 Spotlight 索引 扩展 需要 一 个 继承 自 CSIndexExtensionRequestHandler 的 类 并 实现 以 下 
方法 。 
e searchableIndex:reindexAllSearchableItemsWithAcknowledgementHandler: 
调用 该 方法 以 触发 索引 设备 中 所 有 的 项 。 
e searchableIndex:reindexAllSearchableItemsWithAcknowledgementHandler: 
调用 该 方法 来 触发 验证 指定 唯一 标识 符 的 项 的 有 效 性 。 
实现 一 个 Spotlight 扩展 的 最 佳 实践 与 13.1.3 市 中 所 讨论 的 内 容 相 同 。 


13.4 ”应 用 瘦身 


在 iOS 9 之 前 ， 当 需要 一 个 统一 的 二 进 制 文件 来 支持 多 个 设备 (iPod, iPhone, iPad, LAK 
如 今 的 Apple Watch) 时 ， 你 需要 将 所 有 的 资源 打包 在 一 块 。 这 可 能 会 让 用 户 在 首次 启动 
下 载 资 源 文件 时 感到 不 满 。 

资源 文件 目录 (参见 6.2.3 节 ) 已 被 证 明 是 一 项 非常 有 用 的 功能 ， 不 仅 能 加 快 图 片 加 载 ， 
还 能 帮助 管理 同一 个 图 片 针 对 各 种 设备 的 不 同 版 本 。 然 而 ， 这 也 使 得 用 户 最 终 下 载 的 二 进 
制 文件 变 得 十 分 庞大 。 

另 一 方面 ， 优 化 资源 〈 图 像 、 音 频 剪 辑 、 视 频 剪 辑 ) 的 使 用 意味 着 将 它们 分 为 两 类 。 一 类 
是 总 和 应 用 打包 在 一 块 ， 另 一 类 是 之 后 需要 时 再 从 服务 器 下 载 。 这 个 方法 可 以 减少 一 些 二 
进 制 文件 的 大 小 ， 同 时 避免 了 过 度 使 用 网 络 。 

不 过 ， 这 意味 着 要 编写 很 多 自 定义 代码 ， 并 且 要 维护 服务 器 以 便 在 需要 时 可 以 下 载 资 源 。 
这 对 支持 多 个 皮肤 和 主题 的 应 用 来 说 非常 痛苦 ， 最 受 影 响 的 是 游戏 类 应 用 。 

iOS 9 引入 了 三 个 功能 来 处 理 资 源 分 发 的 相关 问题 : 






































。 分 割 
。 按 需 加 载 资 源 
* Bitcode 


13.4.4. SRJ 


作为 开发 者 ， 你 仍然 可 以 上 传统 一 的 二 进 制 文件 到 Tunes Connect， 包 含 对 所 有 设备 的 可 
执行 程序 和 资源 文件 。App Store 负责 根据 应 用 支持 的 设备 创建 多 个 下 载 包 。 


为 下 载 、 安 装 应 用 的 目标 设备 创建 和 分 发 变种 应 用 包 称 为 分 割 。 
变种 包 包含 特定 的 可 执行 体系 结构 和 目标 设备 需要 的 资源 文件 。 


。 只 下 载 针 对 处 理 器 体系 结构 的 可 执行 文件 。 
。 按 设备 支持 能 力 切 分 GPU 资源 。 
。 根据 设备 类 型 和 分 状 率 切 分 图 像 。 
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App Store 为 安装 了 iOS 9 及 更 新 版 本 的 设备 发 送 分 割 过 的 包 。 对 之 前 的 版 本 
还 是 分 发 统一 的 二 进 制 包 。 分 割 过 的 图 片 必须 位 于 资源 文件 目录 下 。 其 他 位 
置 的 图 片 不 会 被 切 分 。 

















13.4.2 EMAAR 


按 需 加 载 资 源 (http://apple.co/1YrrsmK) 指 的 是 App Store 上 与 下 载 的 应 用 包 分 离 的 应 用 
内 容 。 


从 iOS 9 开始 ， 你 可 以 对 特定 资源 〈 如 图 片 和 音频 剪辑 ) 加 标签 ， 然 后 通过 标签 来 管理 这 
些 资 源 。 

具体 来 说 ， 你 可 以 配置 : 

。 随 应 用 打包 的 资源 

。 应 用 第 一 次 启动 后 安装 的 资源 

。 根据 一 个 关键 词 安装 的 所 有 资源 

。 根据 一 个 关键 词 删除 的 所 有 资源 

这 些 资源 也 可 能 被 分 割 。 这 样 能 保证 资源 总 是 应 用 必需 的 ， 而 不 会 占用 磁盘 空间 。 设 备 上 
包 和 资源 文件 变 小 的 副作用 是 ， 应 用 加 载 时 间 也 会 变 短 。 

例如 ， 如 果 应 用 有 多 个 皮肤 和 主题 ， 那 么 图 片 包 开 始 可 以 只 包含 默认 主题 。 你 也 可 以 在 设 
备 上 保留 一 些 最 常用 的 主题 ,或 者 是 一 个 小 时 前 刚 浏览 过 的 主题 。 


13-12 展示 了 从 App Store 到 设备 按 需 加 载 资源 的 生命 周期 。 


中 的 资源 文件 
应 用 根据 标 
签 请 求 资源 
操作 系统 : 
下 载 资源 包 





















































操作 系统 清 应 用 使 用 资源 包 
理 资源 包 
留存 资源 包 





资源 包 使 用 完毕 。 “| 应 用 根据 标 
资源 包 使 用 完毕 p rri 














图 13-12: 按 需 加 载 资源 的 生命 周期 
你 必须 先 设置 启用 按 需 加 载 资源 的 工程 ， 如 图 13-13 所 示 。 
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四 General Capabilities Resource Tags Info Build Setting: 


PROJECT Basic "m 


Ex HighPerformance 





TARGETS V Assets 

2 HighPerformance Setting a HighPerformance 
[| HighPerformanceTests Embed Asset Packs In Product Bundle No $ 

On Demand Resources Initial Install Tags 

On Demand Resources Prefetch Order 





) shareRead 


(E) documentProvider 


) documentProvider... 





(E) HPContentBlocker 








图 13-13: Xcode 启用 按 需 加 载 资源 设置 


启用 按 需 加 载 资 源 后 ， 下 一 步 是 管理 标签 及 其 关联 的 资源 。 在 Xcode 7 中 ， 工 程 设置 页 包 
含 了 一 个 新 的 Resource Tags 标签 ， 可 用 于 管理 标签 ( 见 图 13-14)。 你 可 以 使 用 资源 文件 
目录 编辑 器 来 关联 标签 和 资源 。 











gO General Capabilities Resource Tags 


PROJECT All Geer! + 


Pa 1 
an HighPerformance Y Initial Install Tags 


TARGETS Y theme default (not built 
A 
十 


:& HighPerformance 
|. ]HighPerformanceTests 
(E) actionRead 

(E ) shareRead 

(E) documentProvider 


(©) documentProvider...  " Prefetched Tag order 


* Download Only On Demand 


| (E) HPContentBlocker 
(E) spotlightIndex Y AR 














图 13-14; Xcode 一 一 Resource Tags 标签 
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图 13-14 中 有 三 类 标签 ， 从 名 字 上 区 分 为 theme default, theme black, theme blue, + 
theme default 标签 添加 到 Initial Install Tags 集合 下 ， 表 明 该 资源 将 会 集成 在 下 载 的 应 用 
包 中 。Prefetched Tag Order 里 没有 标签 ， 这 里 包含 的 是 在 第 一 次 应 用 启动 时 下 载 的 资源 文 
件 。 最 后 ，theme_black [theme blue 放 在 Download Only On Demand 集合 里 ， 表 示 这 些 
资源 只 有 在 应 用 需要 时 才 会 下 载 。 


图 13-15 展示 了 如 何 关联 特定 的 资源 对 象 与 标签 。 














HighPerformance ) 二 Images.xcassets ) 3* settings selected < >: Ho X 
settings selected ee 
Name settings selected © 
Devices 
All & Universal 
iOS | | iPhone 
1x 2x 3x = rad 
OSX Mac 
Universal watchOS | | Apple Watch 
Width Any 
Height Any 


Scale Factors Multiple 


Render As | Default 





On Demand Resource Tags 


theme, blue 
theme black 


theme blue f 





Show Slicing 














13-15: 使 用 资源 文件 目录 编辑 器 关联 标签 


一 旦 完成 设置 ， 一 切 准 备 就 绕 。 最 后 一 步 是 管理 按 需 加 载 资源 的 标签 

使 用 类 NSBundleResourceRequest (Khttp:;//apple.co/lNuyYbs) 管理 按 需 加 载 标 签 的 资 
PEATA 例 13-9 中 的 示例 代码 展示 了 如 何 下 载 或 停止 访问 这 些 资 源 文 件 。 注 
意 ， 为 了 使 用 这 些 资源 文件 ， 你 需要 继续 使 用 与 之 前 相同 的 代码 ， 即 NSBundl 或 


UIImage:imageNamed:, 




















f) 13-9 管理 按 需 加 载 资源 标签 


NSSet *tags = [NSSet setWithArray: @[@"theme_blue"]]; 
NSBundleResourceRequest *req - [[NSBundleResourceRequest alloc] 
initWithTags:tags]; @ 


[req beginAccessingResourcesWithCompletionHandler:^(NSError *e) ( @ 
if(e) ( 
// 错 误 处 理 








} else { 
// 正 常 流程 ,举例 如 下 
UIImage *image = [UIImage imageNamed:@"settings"]; 
} 
H; 


[req conditionallyBeginAccessingResourcesWithCompletionHandler:^(BOOL available) ( © 
if(available) ( O 
// 好 的 ,资源 已 经 可 以 使 用 ,进行 处 理 。 
} else { © 
// 不 可 用 ,也 许 还 未 下 载 或 已 被 清除 ,现在 下 载 。 





} 
HE 


[req endAccessingResources]; © 


Q 为 应 用 希望 使 用 的 标签 创建 一 个 NNBundLeResourceRequest 对 象 。 

O 下 载 请 求 ， 处 理 出 现 错误 或 下 载 成 功 的 场景 。 

O 检查 资源 在 设备 上 是 否 可 用 。 

O 如 果 可 用 ， 那 么 如 前 面 讨 论 的 那样 使 用 。 

© 如 果 不 可 用 ， 使 用 beginAccessingResourcesWithCompletionHan dler: 方法 排队 下 载 。 
Q 通知 系统 你 已 经 完成 使 用 给 定 标签 的 资源 。 












































13.4.3 bitcode 
bitcode 是 一 个 编译 好 的 程序 的 中 间 表 示 形 式 。 


当 应 用 以 bitcode 格式 提交 到 iTunes Connect 后 ， 将 会 在 App Store 中 以 原生 格式 编译 ， 并 
且 链 接 到 最 终 的 二 进 制 文件 。 


使 用 bitcode 允许 人 苹果 公司 在 未 来 对 应 用 的 二 进 制 文件 进行 二 次 优化 ， 而 不 需要 重新 提交 
一 个 新 版 本 到 App Store。 


图 13-16 展示 了 启用 bitcode 选项 的 Xcode 工程 设置 。 新 Xcode 7 工程 的 默认 选项 是 支持 bitcode。 




















PROJECT 


Basic Levels Qy bitcode 


E HighPerformance 


TARGETS Y Build Options 
/? HighPerformance Setting @ HighPerformance 
[^ ]HighPerformanceTests 
E) actionRead 
E) shareRead 


documentProvider 


m 


documentProvider... 


m 


HPContentBlocker 


spotlightlndex 











& 13-16: 启用 bitcode 选项 的 Xcode 工程 设置 
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bitcode 对 iOS 是 可 选 的 ， 但 对 watchOS 是 必须 的 。 


13.5 ”小结 


iOS 9 拥有 一 些 非 常 强大 的 功能 ， 你 可 以 用 它们 来 提高 应 用 的 受 欢 迎 程度 、 底 层 性 能 ， 
以 及 可 感知 的 性 能 。 使 用 通用 链接 提供 一 个 统一 访问 及 可 分 享 的 URL， 不 再 需要 自 定 
SC URL scheme 来 管理 令 人 讨厌 的 跳 转 。 为 应 用 的 公开 内 容 建立 索引 ， 以 便 它们 可 用 于 
Spotlight 搜索 。 理 当 在 应 用 中 使 用 SFSafariViewControLLer， 但 需要 确保 对 iOS 8 或 更 早 的 
版 本 提供 向 下 兼容 。 最 后 ， 应 用 瘦身 是 一 项 必须 启用 的 功能 ， 尤 其 是 当 应 用 的 资源 文件 很 
多 ， 并 且 在 安装 后 并 不 需要 所 有 的 文件 时 ， 你 应 该 按 需 加 载 资 源 。 
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iOS 10 





至 此 ， 原 书 已 经 告 一 段落 。 想 必 读 到 此 处 ， 你 不 免 有 些 意犹未尽 一 一 止步 于 ioOS 9 实在 是 
让 人 心 有 不 甘 。 一 方面 ，iOS 10 已 经 面向 公众 发 布 了 一 段 时 间 ， 截 止 到 2017 年 1 月 4 日 ， 
iOS 10 的 比重 已 经 超过 76%2， 另 一 方面 ，iOS 10 是 当前 最 新 款 的 旗舰 iPhone (iPhone 7 和 
iPhone 7 plus) 唯一 可 用 的 操作 系统 。 作 为 时 代 最 前 沿 的 移动 互联 网 技术 开发 者 ， 我 们 怎 
能 对 iOS 10 视而不见 呢 ? 

与 iOS9 相 比 ，iOS 10 带 来 了 更 多 的 新 特性 ， 开 放 了 更 多 的 扩展 能 力 ， 主 要 体现 在 以 下 方面 : 
。 Siri 扩展 

。 改进 的 通知 

。 VoIP 支持 

。 iMessage 扩展 


这 些 新 特性 赋予 了 第 三 方 应 用 更 强大 的 能 力 ， 也 极 大 地 便利 了 用 户 的 使 用 。 然 而 ， 本 书 的 
主题 是 “高 性 能 ”"。 虽 然 iOS 10 是 当前 最 新 、 最 优秀 的 iOS 版 本 ,但 其 诸多 的 新 特性 却 并 
非 与 “高 性 能 ”主题 密切 相关 。 此 外 ， 本 书 中 介绍 的 诸多 性 能 优化 技巧 通常 是 基于 一 些 最 
佳 实践 和 经 验 总 结 出 来 的 。 因 此 ， 本 书 介绍 的 性 能 方面 的 经 验 和 技巧 绝 非 仅 仅 适用 于 某 一 
个 或 某 几 个 iOS 版 本 。 这 些 技巧 更 像 是 一 棵 常 青 树 ， 即 使 iOS 的 版 本 不 断 地 演变 与 更 迭 ， 
它们 仍然 能 够 永 傈 青 

因此 ， 本 章 介 绍 的 iOS 10 的 相关 知识 并 非 本 书 的 重点 ， 更 多 的 只 是 起 到 索引 的 作用 ， 希望 
能 够 帮助 你 更 快 地 入 门 ， 并 找到 相关 的 学 习 资 料 。 















































ny 


HE 1: 为 完善 本 书 内 容 ， 跟 进 最 新 技术 ， 译 者 创作 了 本 章 。 一 一 编者 注 
注 2: 数据 来 源 于 https//developer.apple.com/support/app-store/ 
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14.1 Siri È 


伴随 着 经 典 的 iPhone 4s, Siri (Speech Interpretation and Recognition Interface, 1&6 4.45 
识别 接口 ) 进入 了 公众 的 视野 中 ， 时 隔 5 4B, SERA KE iOS 10 版 本 中 对 Siri 进行 了 有 
史 以 来 最 大 的 改进 : 通过 开放 扩展 接口 ， 人 允许 第 三 方 开发 者 使 用 。 这 个 扩展 接口 被 称 为 
Sirikit, 

Sirikit 由 两 个 框架 组 成 : Intents 框架 用 于 支持 应 用 和 系统 之 间 的 基础 通信 ，Intents UI 
框架 提供 了 展示 自 定义 用 户 接 口 的 能 
Sirikit 的 工作 原理 如 图 14-1 所 示 : 用 户 “ 输 入 ”语音 ，Siri 的 音频 识别 引擎 根据 词汇 表 识 
别 出 语 音 表 达 的 意图 (Intents)， 然 后 通过 执行 一 系列 的 任务 (动作 )， 产 生 响 应 。 对 于 应 用 
的 开发 人 员 而 言 ， 应 该 关注 词汇 、 应 用 逻辑 和 用 户 界 面 ， 其 他 的 事情 由 Sirikit 为 你 代劳 。 







































































= 



































14-1; SiriKit 的 工作 原理 











就 目前 而 言 ，Sirikit 只 支持 以 下 的 一 些 细 分 领域 ,， 因此， 只 有 服务 于 这 些 领 域 的 应 用 能 
够 利用 Siri 的 特性 进行 开发 。 现 阶段 支持 的 领域 有 : 

。 语音 和 视频 通话 
。 发 送 消 息 
。 收 付款 

。 有 照片 

。 健身 

。 出 行 预定 

。 汽车 控制 命令 〈 限 汽车 供应 商 ) 

e CarPlay 〈 限 汽车 供应 商 ) 

。 餐厅 预定 (需要 苹果 额外 支持 ) 

客观 来 讲 ， 现 阶段 的 Sirikit 能 力 还 比较 有 限 ， 其 扩展 性 和 通用 性 还 不 足以 被 第 三 方 应 用 
广泛 使 用 。 尽 管 如 此 ，Sirikit 仍旧 是 一 个 良好 的 开端 ， 其 前 景 不 可 估量 。 


使 用 Sirikit 的 步骤 如 下 : 
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首先 ， 创 建 Intents 扩展 

在 Xcode 中 打开 应 用 的 工程 

然后 点 击 File > New > Target 

选择 Intents Extension, Sq Ai Next 

输入 相关 的 配置 信息 ， 如 名 称 等 ， 如果 需 要 自 定 义 Siri 的 UI， 不 妨 勾 选 上 include 
UI Extension 选项 

4 最 后 点 击 Finish 


修改 Intents 扩展 中 的 info.plist 文件 
4 配置 IntentsSuported 键 ， 用 于 配置 需要 使 用 哪些 Intents 服务 。 


当前 可 用 Intents 都 隶属 于 前 面 介绍 过 的 那些 领域 。 例 如 ， 要 想 支 持 音频 呼叫 ， 请 使 
用 INStartVideoCaLLIntent。 要 想 了 解 详细 的 Intents 列表 ， 可 以 参见 苹果 的 官方 文 
#4: https://developer.apple.com/library/prerelease/content/documentation/Intents/Conceptual/ 
SiriIntegrationGuide/SiriDomains.html#//apple_ref/doc/uid/TP40016875-CH9-SW2 
h 5 AR 
不 同 的 Intents 会 有 对 应 的 处 理 协议 和 响应 类 ， 例 如 ，INStartVideoCaLLIntent 对 应 的 
处 理 协 议 是 INStartVideoCaLLIntentHandLing， 而 啊 应 类 是 INStartVideoCallIntent- 
Response。 通 过 实现 处 理 协 议和 创建 响应 类 ， 开 发 人 员 可 以 实现 自己 想 要 的 功能 。 以 发 
送信 息 为 例 ， 需 要 实现 INSendMessageIntentHandling 协议 的 方法 ， 并 且 构 造 INSend- 
MessageIntentResponse 类 (或 子 类 ) 的 实例 ， 上 有 具体 见 例 14-1, 


例 14-1 Intents 处 理 逻 辑 


- (void)handleSendMessage:(INSendMessageIntent *)intent completion:(void (^) 
endMessageIntentResponse *response))completion 
(INSendM I R * ))completion { 








* 9 9 9 






































Nr 





















































NSArray<INPerson *» *recipients = intent.recipients; // 获 取 联 系 人 

NSString* userName = recipients[0].displayName; // 这 里 仅 取 第 一 位 联系 人 

NSString *text = intent.content; // 获 取消 息 内 容 

NSUserActivity *userActivity = [[NSUserActivity alloc] initWithActivityType:NSS 
tringFromClass([INSendMessageIntent class])]; 

[userActivity addUserInfoEntriesFromDictionary:@{@"UserName" : userName, 
"Message" : text}]; 


INSendMessageIntentResponse *response - [[INSendMessageIntentResponse alloc] in 
itWithCode:INSendMessageIntentResponseCodeSuccess userActivity:userActivity]; 
completion(response); 


} 
4 Siri 识别 了 用 户 的 语音 ， 并 识别 出 用 户 要 使 用 该 应 用 发 送 消 息 时 ， 上 述 代码 就 会 被 
触发 执行 ， 然 后 启动 该 应 用 。 应 用 会 通过 NsuserActivity 获取 Siri 扩展 所 传递 的 数据 : 
即 用 户 名 和 消息 内 容 。 
向 用 户 申 请 使 用 Siri 的 权限 
要 想 使 用 Sirikit， 还 需要 在 应 用 中 执行 申请 授权 。 你 可 以 在 应 用 运行 的 任意 时 刻 调用 
INPreferences 类 的 requestSiriAuthorization: 方法 ， 从 而 申请 使 用 Siri 的 权限 。 
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AK Sirikit 的 更 多 信息 ， 可 以 参见 苹果 公司 提供 的 官方 文档 : https://developer.apple.com/ 
library/prerelease/content/documentation/Intents/Conceptual/SiriIntegrationGuide/index.html#// 
apple_ref/doc/uid/TP40016875-CH11-SW1， 以 及 全 球 开发 者 大 会 中 对 应 的 教程 : https:// 
developer.apple.com/videos/play/wwdc2016/217/。 


14.2 ”改进 的 通知 


通知 系统 一 直 都 是 iOS 提供 的 核心 能 力 ， 在 iOS 10 版 本 中 ,苹果 对 其 作出 了 重大 的 改进 
主要 体现 在 以 下 几 个 方面 : 


(1) 更 加 丰富 的 展示 形态 
(2) 自 定义 的 交互 事件 
(3) 改进 的 API 一 一 新 API 固然 好 ， 但 大 量 的 旧 API 被 标记 为 deprecated 


14.2.1 ”申请 权限 


首先 ， 需 要 申请 使 用 通知 的 权限 ， 调 用 UNUserNotificationCenter 的 requestAuthorizatio 
nWithOptions:completionHandler: 方法 即 可 ， 且 体 见 例 14-2, 


例 14-2 注册 申请 使 用 通知 的 权限 
[[UNUserNotificationCenter currentNotificationCenter] 
requestAuthorizationWithOptions: 
UNAuthorizationOptionBadge | 
UNAuthorizationOptionAlert | 
UNAuthorizationOptionSound 
completionHandler:^(BOOL granted, NSError * _Nullable error) { 
H; 


14.2.2 ”触发 器 


接 下 来 介绍 的 是 iOS 10 中 引入 的 触发 器 (Triger)。 触 发 器 用 于 实现 在 特定 条 件 下 触发 通 
知 。 系 统 提供 了 以 下 4 TRASTERO as, ILE 14-2, 



















































































14-2: iOS 10 支持 的 4 种 触发 器 


。 Push; 支持 远程 推送 
。 时 间 间 隔 : 定时 触发 ， 支 持 多 次 重复 触发 
。 HD: 在 某 个 时 间 点 进行 触发 





。 定位 服务 : 当 用 户 进入 或 离开 某 一 区 域 时 触发 
触发 器 使 得 发 送 通知 变 得 格外 简单 ， 例 14-3 演示 了 如 何 使 用 时 间 间 隔 触发 通知 


例 14-3 使 用 触发 器 推送 通知 





NSString* categoryIdentifier = @"HelloNotificationID"; 





// 创 建 要 推送 的 内 容 


UNMutableNotificationContent *content = 


init]; 


content.title = @"i0S 10 Notification"; 
content.subtitle = @"Trigger by Time Interval"; 
content.body = @"Hello Notification"; 


content.categoryIdentifier 





= categoryIdentifier; 


// 创 建 触发 器 :1 秒 之 后 触发 ,不 重复 


UNTimeIntervalNotificationTrigger *timeIntervalTrigger 


nTrigger 


triggerWithTimeInterval:1 repeats:NO]; 


UNNotificationRequest *request - 


[UNNotificationRequest 


0 


[[UNMutableNotificationContent alloc] 


= [UNTimeIntervalNotificatio 


requestWithIdentifier:notificationIdentifier content:content trigger:trigger]; 
[[UNUserNotificationCenter currentNotificationCenter] 
addNotificationRequest:request 

withCompletionHandler:^(NSError * _Nullable error) { 


// 完 成 推送 
Hs 


@ 这 里 的 categoryIdentifier 很 重要 ， 后 面 还 会 对 其 进行 介绍 





iOS 10 的 通知 销 息 中 还 可 以 携带 附件 ， 目 前 可 以 携带 图 片 、 音 频 和 视频 ， 但 附件 有 大 小 


限制 : 








(D 图 片 ， 包 括 JPEG, GIF, PNG, 最 大 为 10M 








(2) 音频 ， 包 括 AIFF, MP3, WAV, MPEG4 音频 ， 
(3) 视频 ， 包 括 MPEG, MPEG2, MPEG4, AVI, 





最 大 为 SM 
最 大 为 30M 


具体 做 法 是 设置 UNMutableNotificationContent 的 attachments 属性 ， 代 码 见 例 14-4, 


il 14-4 在 通知 中 使 用 附件 


NSURL *fileURL = [[NSBundle mainBundle] 
URLForResource:Q"picture" withExtension: @"png" ]; 

UNNotificationAttachment *attachment - [UNNotificationAttachment 
attachmentWithIdentifier:@"aAttachment" URL:fileURL options:nil error:nil]; 

content.attachments = [attachment]; 








SR 





这 里 要 强调 一 点 ，fileURL 只 支持 本 地 文件 URL， 要 想 展示 服务 器 端的 图 片 或 播放 远程 的 
音频 和 视频 ， 我 们 还 需要 其 他 的 方法 ， 后 面 会 对 此 进行 详细 介绍 














14.2.3 ”为 通知 添加 交互 


通过 定义 和 注册 动作 (action)， 











你 可 以 为 通知 的 交互 展示 界 国 








ij 添加 交互 效果 。 有 具体 做 法 是 





创建 UNNotificationCategory， 然 后 向 其 中 添加 UNNotificationAction 类 的 实例 ， 最 后 将 
category 注册 到 UNUserNotificationCenter 中 ， 有 具体 代码 见 例 14-5, 
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Bil 14-5 注册 通知 交互 

UNNotificationAction *action = [UNNotificationAction 
actionWithIdentifier:Q"com.app.more" title:@" 查 看 更 多 " 
options:UNNotificationActionOptionForeground]; @ 

NSString* notificationIdentifier = @"HelloNotificationID"; @ 

UNNotificationCategory *category = [UNNotificationCategory 
categoryWithIdentifier:notificationIdentifier actions:@[action] 
intentIdentifiers:Q[] 
options:UNNotificationCategoryOptionCustomDismissAction]; © 

[[UNUserNotificationCenter currentNotificationCenter] 
setNotificationCategories:[NSSet setWithArray:Q[category]]]; @ 


@ 每 个 action 代表 一 个 可 用 于 交互 的 UI 元 素 。 本 例 是 一 个 按钮 ， 也 可 以 是 一 个 文本 
(UNTextInputNotificationAction) 。 

@ category 的 标识 符 ， 应 与 通知 内 容 的 标识 符 一 致 〈 见 例 14-3), 

Q 每 个 category 可 以 包含 一 组 action， 以 对 应 某 一 种 通知 的 多 个 UI 元素 。 

@ 将 category 注册 到 UNUserNotificationCenter 中 。 


已 经 为 通知 添加 了 交互 元 素 (按钮 和 文本 框 )， 显 然 还 需要 处 理 交 互 的 逻辑 。 在 发 生 交 互 
上 时， 会 通过 UNUserNotificationCenter 的 委托 UNUserNotificationCenterDelegate 进行 通 
知 。 见 例 14-6。 


例 14-6 处 理 交 互 逻辑 
#pragma mark - UNUserNotificationCenterDelegate 
- (void)userNotificationCenter:(UNUserNotificationCenter *)center 
didReceiveNotificationResponse:(UNNotificationResponse *)response 
withCompletionHandler:(void(^)())completionHandler 





{ 

[[NSNotificationCenter defaultCenter ] 
postNotificationName:@"didReceiveNotificationResponse" 
object :nil 
userInfo:@{ @"actionID" : response.actionIdentifier}]; 

completionHandler(); 

} 


注意 ， 这 里 的 委托 方法 的 参数 是 UNNotificationResponse， 通 过 它 可 以 拿 到 很 多 东西 。 

















14-3; UNNotificationResponse 的 数据 结构 


14.24 ”完全 自 定 义 展示 通知 
你 是 否 已 经 厌倦 了 千篇一律 的 通知 界面 ， 而 想 要 开发 一 个 个 性 十 足 的 应 用 ?苹果 公司 在 
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iOS 10 版 本 中 提供 了 这 方面 的 能 力 。 拿 售卖 火车 票 的 应 用 来 说 ， 如 有 果 能 在 出 票 成 功 后 向 用 
户 推送 如 图 14-4 所 示 的 效果 ， 那 用 户 体验 一 定 很 不 错 。 








12 车 4 号 上 铺 


添加 至 日 历 | 


保存 到 相册 


| 查看 订单 | 











图 14-4: 自 定义 通知 展示 效果 


这 里 需要 强调 的 是 ， 所 谓 的 自 定义 界面 ， 其 实 还 是 有 比较 大 的 限制 : 不 可 以 交互 ， 也 不 接 
受 触摸 事件 ， 仅 仅 用 于 信息 的 展示 。 

为 实现 自 定 义 的 界面 ，iOS 10 提供 了 新 的 扩展 。 通 过 开发 通知 扩展 ， 第 三 方 的 开发 人 员 才 
可 以 实现 通知 的 自 定义 展示 方式 ， 具 体 做 法 如 下 : 

。 在 Xcode 中 打开 工程 ， 点 击 File > New > Target, Wł% Notification Content Extension 
。 输入 产品 名 称 等 必要 的 信息 ， 然 后 点 击 Finish 

。 修改 生成 的 info.plist 文件 


UNNotificationExtensionCategory 键 用 于 指定 通知 的 categoryIdentifier (不 妨 回忆 一 下 
例 14-3) ， 从 而 将 扩展 与 某 个 类 别 的 通知 关联 起 来 


当 收 到 通知 时 ， 会 触发 UNNotificationContentExtension 的 didReceiveNotification: 方 
法 。 有 具体 代码 见 例 14-7。 


例 14-7 ”响应 通知 ， 展 示 UI 
- (void)didReceiveNotification:(UNNotification *)notification { 


// 将 通知 的 数据 展示 在 UI 之 上 


self.label.text = notification.request.content.body; 


} 
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如 果 通 知 对 应 的 category 还 指定 了 动作 ， 那 么 还 可 以 通过 didReceiveNotificationRespons 
e:completionHandler: 方法 拦截 用 户 交 互 ， 参 见 例 14-8。 


例 14-8 响应 用 户 操作 
- (void)didReceiveNotificationResponse:(UNNotificationResponse *)response 


completionHandler: 
(void (^)(UNNotificationContentExtensionResponseOption))completion 








// 在 此 处 编写 代码 ,响应 用 户 的 操作 


completion(UNNotificationContentExtensionResponseOptionDoNotDismiss); 





j 


14.2.5 ”通知 服务 扩展 

到 目前 为 止 ， 我们 已 经 介绍 了 通知 系统 的 主要 内 容 ， 陪 明 的 你 一 定 会 发 现 一 个 问题 ， 整个 
通知 系统 在 能 力 方面 似乎 并 不 完备 。 通 知 携带 的 附件 信息 只 支持 访问 本 地 的 资源 。 这 意味 
着 其 表达 能 力 其 实 非常 的 匮乏 。 毕 竟 打 包 在 应 用 中 的 资源 文件 既 不 丰富 ， 也 不 灵活 ， 很 难 
实现 向 用 户 推送 丰富 多 彩 的 信息 。 举 个 例子 ， 要 想 向 用 户 推送 一 部 新 电影 的 宣传 片 ， 却 发 
现 通知 系统 要 求 相关 的 视频 必须 存 于 用 户 的 手机 本 地 ， 多 么 滑稽 啊 ! 

幸运 的 是 ， 侠 果 提 供 了 有 效 的 解决 方案 ， 人 允许 第 三 方 开发 者 通过 通知 服务 扩展 来 实现 类 似 
的 需求 。 通 知 服务 扩展 为 第 三 方 的 开发 者 提供 了 能 力 ， 从 而 能 够 将 推送 的 消息 进行 加 工 处 
里 (如 下 载 资源 )， 具 体 做 法 如 下 。 

。 在 Xcode 中 打开 工程 ， 点 击 File» New» Target, Wf% Notification Service Extension, 
。 输入 产品 名 称 等 必要 的 信息 ， 然 后 点 击 Finish, 

与 发 送 远程 Push 的 服务 端 开发 人 员 约 定好 ， 将 推送 的 信息 标记 为 “可 变 ”("mutable- 
content" = 1)， 见 例 14-9, 


例 14-9 ”服务 端 推送 消息 时 ， 标 记 为 可 变 




































































xd 














{ 
"aps" : { 
"alert" : { 
"title" : "iOS 10 Push Notification", 
"subtitle" : "Trigger by Push", 
"body" : "Hello Notification" 
J 
"category" : "HelloNotificationID", 
"mutable-content" : 1 
f; 
"my-attachment" : "https://example.com/photo. jpg" 
} 


然后 ， 在 扩展 中 对 服务 端 推送 的 消息 内 容 进行 修改 。 下 载 其 中 对 应 的 远程 资源 ， 使 之 成 为 
本 地 资源 ， 参 见 例 14-10。 


例 14-10 修改 推送 消息 并 下 载 远 程 资 源 


@interface NotificationService () 
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@property (nonatomic, strong) void (^contentHandler)(UNNotificationContent 
*contentToDeliver); 

(property (nonatomic, strong) UNMutableNotificationContent *bestAttemptContent; 
(property (nonatomic, strong) NSURLSessionDownloadTask *downloadTask; 


(gend 
(implementation NotificationService 


- (void)didReceiveNotificationRequest: (UNNotificationRequest *)request 
withContentHandler: (void (^)(UNNotificationContent * _Nonnull))contentHandler { 
self.contentHandler = contentHandler; 
self.bestAttemptContent = [request.content mutableCopy]; 
NSString *URLString = request.content.userInfo[Q"my-attachment"]; 
NSURL* resourceURL = [NSURL URLWithString: URLString]; 


self .downloadTask = [[NSURLSession sharedSession ] 
downloadTaskWithURL:resourceURL 
completionHandler:^(NSURL * _Nullable location, 
NSURLResponse * _Nullable response, 
NSError * _Nullable error) { 
NSURL *savedURL = 基 个 可 用 的 本 地 文件 URL; 
[[NSFileManager defaultManager] moveItemAtURL:location toURL:savedURL 
error:nil]; 
NSError *attachmentError - nil; 
NSString *identifier - [NSString stringWithFormat: 
@"%@%@", URLString, [NSDate date]]; 
UNNotificationAttachment *attachment - [UNNotificationAttachment 
attachmentWithIdentifier: identifier 
URL: savedURL 
options:nil 
error: \&attachmentError]; 
if (attachmentError) { 
self.bestAttemptContent.body = [NSString stringWithFormat:@"%@ 
attachmentError:%@", self.bestAttemptContent.body, attachmentError. 
localizedDescription]; 
) else { 
// 将 下 载 完成 的 文件 添加 到 content 
self.bestAttemptContent.attachments = Q[attachment]; 














LL 

















} 
self.contentHandler(self.bestAttemptContent); 
H; 
[self.downloadTask resume]; 
} 
- (void)serviceExtensionTimeWillExpire { 
// 无 法 在 规定 的 时 间 内 完成 下 载 ,取消 下 载 
[self.downloadTask cancel]; 
self.contentHandler(self.bestAttemptContent); 
} 
@end 
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苹果 对 通知 服务 扩展 有 一 些 限 制 ; 
。 扩展 执行 的 时 间 不 得 超过 30 秒 ， 因 此 ， 远 程 资 源 必须 在 30 秒 钟 之 内 下 载 完毕 ， 否 则 会 
先 触发 serviceExtensionTimeWillExpire 方 法， 然后 中 止 扩 展 的 执行 。 
要 谨慎 使 用 远程 推送 的 附件 ， 一 定 要 避免 因为 数据 量 过 大 而 导致 用 户 的 数据 流量 费用 
剧 增 。 你 可 以 使 用 第 3 章 中 介绍 的 知识 点 ， 根 据 用 户 的 实际 网 络 状 况 (WiFi、4G 或 
其 他 )， 灵 活 地 调整 资源 下 载 的 逻辑 。 例 如 ， 在 使 用 WiFi 网 络 时 下 载 品质 更 高 的 音 
频 、 视 频 或 图 片 资 源 ， 而 在 蜂窝 网 络 时 不 下 载 或 下 载 品 质 较 低 的 资源 。 


14.3 iMessage 扩 展 


iMessage 是 苹果 公司 在 2011 年 随 iOS 5 推出 的 一 项 免费 服务 ， 用 户 可 以 通过 它 利 用 互联 网 
服务 发 送 文字 信息 和 图 片 信息 。 时 隔 5 年 ， 伴 随 着 IOS 10 的 出 现 ， 这 一 服务 终于 得 到 了 重 
大 更 新 。 

通过 提供 iMessage 的 扩展 ， 苹 果 人 允许 第 三 方 对 iMessage 进行 扩展 ， 这 使 得 iMessage 有 了 
无 限 的 可 能 性 。 官 方 提供 了 以 下 的 扩展 类 型 : 

。 贴纸 (表情 包 ) 

。 交互 式 信息 

。 其 他 内 容 一 一 照片 、 视 频 、 文 本 、 链 接 ， 等 等 

同时 ，iOS 10 还 提供 了 类 似 App Store 的 商店 功能 ， 即 “Messages App Store”, HJ? "TER 
过 这 个 商店 对 扩展 进行 管理 。 

iOS 提供 了 一 个 全 新 的 Messages Framework 框架 ， 通 过 它 可 以 开发 信息 扩展 应 用 ， 标 准 的 
步骤 如 下 。 

。 创建 信息 扩展 

苹果 就 此 提供 了 两 种 选择 一 种 方案 是 仅仅 创建 独立 的 扩展 ， 用 于 在 “Messages App 
Store” 中 发 售 ， 另 一 种 方案 是 为 现 有 的 应 用 添加 一 个 扩展 包 ， 则 该 扩展 会 随 应 用 进行 
发 布 。 

对 于 消息 扩展 本 身 而 言 ， 两 者 没有 本 质 区 别 。 在 Xcode 中 ， 点 击 File > New > Project, 


选择 iMessage Application; 或 者 点 击 File > New > Target, Wł% iMessage Extension, 











I 























。 编写 消息 扩展 的 根 ViewControler 
消息 扩展 的 根 ViewController 继承 于 MSMessagesAppViewController, MSMessagesAppView- 
Controller 本 身 也 继承 于 UIViewController， 所 以 使 用 起 来 非常 轻松 。 
如 果 只 是 想 要 编写 一 个 发 送 贴纸 的 扩展 ， 则 可 以 直接 使 用 MSStickerBrowserView 类 。 它 能 
够 以 网 格 的 形式 展示 贴纸 ， 还 支持 翻 页 ， 参 见 例 14-11。 


例 14-11 使 用 MSStickerBrowserView 


@interface MessagesViewController ()<MSStickerBrowserViewDataSource> 














@property (nonatomic) MSStickerBrowserView* browserView; 
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(property (nonatomic) NSArray<MSSticker*>* stickers; 
(gend 
@implementation MessagesViewController 


- (void)viewDidLoad { 
[super viewDidLoad]; 


self.stickers = 载 入 本 地 贴纸 资源 ,构造 NSticker 对 象 数组 
self.browserView = [[MSStickerBrowserView alloc] init]; 
self.browserView.dataSource - self; 


[self.view addSubview: self.browserView]; 
self.browserView.backgroundColor - [UIColor lightGrayColor]; 
} 


|#pragma mark - MSStickerBrowserViewDataSource 


- (NSInteger)numberOfStickersInStickerBrowserView: 
(MSStickerBrowserView *)stickerBrowserView { 
return [self.stickers count]; 


} 


- (MSSticker *)stickerBrowserView: (MSStickerBrowserView *)stickerBrowserView 
stickerAtIndex:(NSInteger)index { 
return self.stickers[index]; 


} 


MSStickerBrowserView 的 用 法 其 实 大 有 似曾相识 之 感 ，MSStickerBrowserViewDataSource JH 
于 向 MSStickerBrowserView 提供 数据 源 ， 其 用 法 与 UITableView 格外 相似 。 


此 外 ，MSMessagesAppViewController 还 在 UIViewController 的 基础 上 增加 了 一 些 特 有 
的 生命 周期 事件 和 一 些 常用 的 属性 。 其 中 ， 最 重要 的 要 数 @property MSConversation 


*activeConversation;, 


MSConversation 用 于 表示 一 次 对 话 ， 调 用 MSConversation 可 以 发 送 消 息 ， 具 体 API 如 下 。 


// 会 话 中 插入 MSMessage 

- (void)insertMessage:(MSMessage *)message completionHandler:(nullable void (^) 
(NSError * Nullable))completionHandler; 

// 会 话 中 插入 发 送 贴纸 

- (void)insertSticker:(MSSticker *)sticker completionHandler:(nullable void (^) 
(NSError * Nullable))completionHandler; 

// 会 话 中 插入 发 送 文本 消息 
- (void)insertText:(NSString *)text completionHandler:(nullable void (^)(NSError * 
_Nullable))completionHandler; 

// 会 话 中 插入 附件 

- (void)insertAttachment:(NSURL *)URL withAlternateFilename:(nullable NSString 
*)filename completionHandler:(nullable void (^)(NSError * _Nullable)) 
completionHandler; 


苹果 公司 并 没有 将 信息 扩展 简单 地 定义 为 发 送 表 情 包 。 通 过 定义 MsSMessage， 你 能 够 实现 更 
加 复杂 的 交互 式 消息 ， 参 见 例 14-12。 
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f| 14-12 使 用 MSMessage 


MSMessageTemplateLayout* layout = [[MSMessageTemplateLayout alloc] init]; 
layout.imageTitle = @" 图 片 标题 "; 

layout.caption = @" 标 题 "; 

layout.image = [UIImage imageNamed:Q"someImage"]; 


MSMessage* msg - [[MSMessage alloc] init]; 
msg.layout = layout; @ 


[self.activeConversation insertMessage:msg completionHandler:^(NSError * _Nullable 
error) ( 

NSLog(@"%@", error); 
H; 











@ MSMesaage 的 layout 属性 决定 了 消息 的 展示 布局 ， 目 前 只 支持 一 种 布局 ， 即 
MSMessageTemplateLayout， 其 展示 效果 如 图 14-5 所 示 。 















































图 片 ， 音 频 或 视频 
图 片 标题 
图 片 副 标题 


主 标题 说 明 右 对 齐 主 标题 说 明 
副标题 说 明 右 对 齐 副标题 说 明 








14-5; MSMessageTemplateLayout 的 展示 布局 





关于 Messages 框架 的 详细 信息 ， 可 以 参见 苹果 的 官方 文档 *， 以 及 2016 年 的 全 球 开发 者 大 
会 中 的 相关 视频 *。 此 外 , 苹果 公司 还 提供 了 一 个 完整 的 示例 , 演示 了 利用 Messages 框架 与 
好 友 合 作 设 计 一 款 冰 淇 淋 ”。 


























Messages: https://developer.apple.com/reference/messages. 
iMessage Apps and Stickers, Part 1: https://developer.apple.com/videos/play/wwdc2016/204/; Part 2: https:// 
developer.apple.com/videos/play/wwdc2016/224/. 


: Ice Cream Builder: A simple Messages app extension: https://developer.apple.com/library/prerelease/content/ 


samplecode/IceCreamBuilder/Introduction/Intro.html. 





14.4 VoIPx $E 


iOS 10 提供 了 一 套 新 的 框架 一 一 CaLLKit。 利 用 CaLLKit， 第 三 方 的 即时 通信 软件 可 以 得 到 
与 系统 电话 应 用 一 致 的 体验 。 这 对 于 运营 商 来 说 无 疑 是 个 沉重 的 打击 。Callkit 提供 的 能 
力 如 下 。 


。 通知 能 力 : 在 通话 发 起 时 ， 接 收 方 可 以 收 到 通知 。 这 个 通知 是 全 屏 展 示 且 伴 有 铃声 ， 体 
验 与 系统 内 置 的 电话 服务 一 致 。 

。 使 第 三 方 语音 电话 拥有 更 高 优先 级 ， 通 话 时 不 被 系统 来 电 打 断 。 

。 通话 记录 : 通话 过 程 也 会 记录 在 系统 电话 应 用 的 最 近 通 话 中 。 

。 通话 拦截 : 在 来 电 时 ， 对 来 电 进 行 阻止 或 识别 。 


这 里 需要 强调 一 点 ，Callkit 的 定位 并 不 是 提供 通信 服务 ， 而 是 提供 系统 的 入 口 。 真 正 的 
通信 协议 、 通 信 过 程 需要 第 三 方 应 用 的 开发 人 员 自 己 来 实现 。 第 三 方 应 用 的 开发 人 员 可 
以 通过 调用 Callkit 的 API， 将 通信 过 程 反映 到 系统 的 电话 应 用 中 。 理 解 这 一 点 后 ， 再 
学 习 使 用 Callkit MAA DIRS. Callkit 有 两 个 非常 重要 的 类 ， 分 别 是 CXProvider 和 
CXCallController, 


cxprovider 用 于 通知 ， 进 而 更 新 系统 的 状态 。CattKtt 的 核心 就 是 将 第 三 方 系统 的 通信 状 

态 报告 给 系统 。CXProvider 有 一 系列 以 reportXXX 开头 的 方法 ， 用 于 通知 系统 当前 通信 的 

状态 : 

。 reportNewIncomingCallWithUUID:update:completion: 报告 电话 呼 入 状态 

e reportOutgoingCallWithUUID:startedConnectingAtDate: 报告 电话 (连接 中 ) 状态 

。 reportOutgoingCallWithUUID:connectedAtDate: 报告 电话 (已 连接 ) 状态 

e reportCallWithUUID:updated: 报告 电话 更 新 状态 (是 否 支 持 视频 、 是 否 进行 群 组 通话 、 
是 否 挂 起 并 保持 通话 等 ) 

e reportCallWithUUID:endedAtDate:reason: 报告 电话 挂 断 状 态 

举例 ， 当 应 用 收 到 语音 /视频 呼 入 请 求 时 ， 调 用 reportNewIncomingCallWithUUID:update:c 

ompletion: 方法 ， 系 统 会 展示 出 有 电话 呼 入 的 界面 。 


对 于 唤起 的 通话 界面 ， 用 户 可 以 进行 一 些 操作 ， 比 如 点 击 “ 挂 断 ” 按 钮 。 这 些 操作 可 以 
通过 CXProvider 的 委托 CXProviderDelegate 进行 拦截 处 理 。 例 如 ， 当 用 户 选择 接听 电 
话 时 ， 会 触发 provider:performAnswerCallAction: 方法 ， 而 在 用 户 挂 断 电 话 时 ， 会 触发 
provider:performEndCallaction: 方法 。 


CxCaLLController 用 于 发 起 相关 的 动作 ， 其 中 包括， 呼叫、 结束 呼叫 、 保 留 等 。 
CXCallController 会 通过 CXTransaction 发 起 不 同 的 动作 ， 每 种 动作 分 别 对 应 不 同 的 action, 
例如 : 

e CXAnswerCallaction; 接听 电话 

e CXStartCallAction; 开始 外 呼 电 话 

e CXEndCallAction; 结束 通话 

e CXSetHeldCallaction. 保持 通话 
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看 到 这 里 ， 你 可 能 会 觉得 一 头 雾 水 。 如 果 通 过 CXCallController 发 起 了 通话 ， 接 收 方 是 如 

何 收 到 通知 的 呢 ? Eee Callkit 并 不 参与 通信 相关 的 事情 。 

事实 上 ， 你 需要 使 用 PushKit， 通 过 Push 通道 发 送 和 接收 通话 (来 电 ) 通知 。 而 具体 到 通 

信 过 程 ， 需 要 开发 人 员 开 发 网 络 服务 ， 并 且 调 用 AvFoundation 对 音频 或 视频 进行 采集 和 回 

放 ， 最 终 完 成 整个 通信 的 闭环 。 
一 定 要 谨慎 地 使 用 3 数据 流 量 ， 要 避免 因 为 数据 量 过 大 而 导致 用 F 的 数据 流量 ER 费用 剧 
增 。 你 可 以 使 用 第 3 章 中 介绍 的 知识 点 ， 根 据 用 户 的 实际 网 络 状 况 (WiFi、4G 或 其 
他 )， 及 时 调整 通信 音频 、 视 频 质 量 ， 在 用 户 流量 开销 和 服务 质量 之 间 进 行 权 衡 。 


关于 Callkit 的 更 多 知识 ， 可 以 参见 苹果 提供 的 示例 “。 

















iE 6: Speakerbox: Using CallKit to create a VoIP app https://developer.apple.com/library/prerelease/content/ 
samplecode/Speakerbox/Introduction/Intro.html 





作者 介绍 

Gaurav Vaish 在 12 岁 时 首次 接触 到 GW-BASIC， 然 后 就 因为 它 的 简洁 而 爱 上 了 它 。 在 随 
后 的 20 多 年 中 ， 他 使 用 过 大 多 数 的 主流 语言 ， 在 所 有 流行 的 操作 系统 上 ， 甚 至 在 每 一 种 
受 欢 迎 的 设备 上 都 编写 过 代码 。 

他 现在 就 职 于 雅虎 总 部 的 移动 和 新 兴 产 品 团队 一 一 具体 来 说 是 移动 SDK 小 组 。 这 个 小 组 
的 使 命 是 创建 优化 的 可 重用 方案 ， 将 其 整合 到 雅虎 的 移动 应 用 中 ， 它 们 能 够 在 几 十 种 设备 
上 运行 ， 每 月 有 数 亿 用 户 使 用 ， 每 周 执行 超过 十 亿 次 的 用 户 交 互 ， 并 且 每 天 处 理 超过 十 亿 
次 的 网 络 连接 。 

Gaurav 于 2002 年 在 Adobe Systems India 开启 了 他 的 职业 生涯 ， 就 职 于 工程 解决 方案 部 门 。 
2005 年 ， 他 成 立 了 自己 的 公司 一 一 Edujini 实验 室 ， 专 注 于 企业 培训 和 协作 学 习 。 


Gaurav 获 有 印度 IIT Kanpur 电子 工程 语音 信号 处 理 专业 的 科技 学 士 学 位 。 


他 著 有 Reflections by IITians 和 Getting Started with NoSOL， 还 管理 着 自己 的 私人 博客 http:// 
www.mlÜv.com, 


封面 介绍 

本 书 封面 中 的 动物 是 一 只 中 贼 鸭 ， 它 们 是 一 种 广泛 出 现在 世界 各 地 的 迁徙 海岛 。 它 们 在 热 
带 海洋 过 冬 ， 然 后 在 夏天 返回 北部 ， 在 北极 营 原 产 蛋 。 虽 然 它 们 的 名 字 与 波 美 拉 尼 亚 的 波 
罗 的 海地 区 无 关 ， 但 波 美 拉 尼 亚 贼 鸥 是 这 种 鸟 常常 被 叫 错 的 称谓 。 

AUF SP US A KA 45 厘米 至 66 厘米 之 间 ， 体 重 近 1 公斤 。 因 为 它们 与 短 尾 贼 鸭 〈 另 一 
种 海岛 ) 十 分 相似 ， 所 以 想 要 辨别 该 种 贼 欧 会 较为 困难 。 实 际 上 ， 成 年 的 中 贼 鸭 有 多 种 形 
态 ， 或 者 具有 三 种 不 同 的 颜色 图 娄 。 这 三 种 图 案 包 含 棕色 、 黑 色 和 白色 的 阴影 ， 但 通常 有 
白色 的 下 腹部 和 白色 起 膀 。 

中 贼 殉 以 鱼 、 腐 内、 小 鸟 ， 划 至 咕 凌 动物 为 食 。 它 们 会 在 飞行 途中 从 海 鸣 、 惹 鸥 或 塘 狗 那 
偷 鱼 ， 只 会 被 成 年 的 白 尾 唐 和 金鹰 猎 食 。 一 旦 峻 性 中 贱 欧 在 北极 筑 梨 ， 它 们 会 在 地 面 的 草 
集中 孵育 两 三 颗 黄 褐色 的 鸟 蛋 。 中 贼 鸭 以 护 时 性 强 而 著称 ; 虽然 它们 不 能 造成 较 大 的 伤 
害 ， 但 若 有 一 个 生气 的 鸟 妈 妈 俯 冲 至 你 的 头 部 ， 则 确实 是 个 重 梦 啊 ! 

O'Reilly 封面 上 的 许多 动物 都 已 濒临 灭绝 ， 但 它们 的 存在 对 世界 至 关 重 要 。 想 要 了 解 如 何 
帮助 它们 ， 可 以 登录 animals.oreilly.com。 


本 书 封面 图 片 来 自 Lydekker 的 Royal Natural History 一 书 。 
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RE 展 阅 dx 


全 彩印 刷 


。 出 自 国际 知名 的 设计 心理 学 博士 Susan Weinschenk 之 手 
。 ”以 创造 美观 实用 的 设计 为 宗 央 ， 讨 论 设计 师 必须 了 解 的 心理 学 问题 
。 每 个 问题 都 配 以 权威 经 典 示例 ， 并 给 出 即 学 即 用 的 设计 建议 
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高 性 能 iOS 应 用 开发 


本 书 为 有 经 验 的 iOS 开 发 者 提供 构建 优异 应 用 移动 性 能 所 需 的 开发 建议 
和 最 佳 实践 ， 帮 助 读 者 解决 常见 性 能 问题 。 


作者 Gaurav Vaish 从 工程 角度 演示 了 编写 最 佳 代码 的 方法 ， 详 尽 介 绍 如 何 
设计 和 优化 i05 应 用 ， 以 便 在 网 络 较 差 、 内 存 较 低 的 情况 下 提供 流畅 的 
用 户 体 验 。 书 中 还 提供 了 可 以 反复 使 用 的 Objective-C 代 码 ， 以 及 一 些 能 
够 从 众多 应 用 中 脱颖而出 的 高 性 能 原生 iOS 应 用 。 


m 概述 跟踪 应 用 性 能 时 需要 衡量 的 参数 以 及 如 何 衡量 性 能 。 


m 通过 最 小 化 内 存 和 功 耗 以 及 并 发 编程 来 编写 高 效应 用 ， 并 探索 一 
些 相关 选项 。 


m 优化 应 用 的 生命 周期 和 Ul， 以 及 网 络 、 数 据 共享 和 安全 功能 。 
m 了 解 应 用 的 测试 、 调 试 和 分 析 工 具 ， 并 监控 应 用 。 


m 从 真实 用 户 处 收集 数据 来 分 析 应 用 的 使 用 情况 ， 找 出 瓶颈 并 修 
复 。 


Gaurav Vaish 就 职 于 雅虎 公司 的 移动 和 新 兴 产 品 团队 ， 为 每 月 有 数 亿 
人 使 用 的 移动 应 用 创建 优雅 的 可 重用 方案 。 他 曾 是 IIT 全 球 指导 计划 
的 成 员 ， 还 在 印度 班加罗尔 创立 了 InColeg Learning 及 Edujini Labs 有 限 
公司 。 
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看 完了 


如 果 您 对 本 书 内 容 有 疑问 ， 可 发 邮件 至 contact@turingbook.com， 会 有 编辑 
或 作 译 者 协助 答疑 。 也 可 访问 图 灵 社 区 ， 参 与 本 书 讨 论 。 


如 果 是 有 关 电 子 书 的 建议 或 问题 ， 请 联系 专用 客服 邮箱 : 


ebook@turingbook.com, 
在 这 可 以 找到 我 们 : 


微 博 @ 图 灵 教育 : 好 书 、 活 动 每 日 播报 

微 博 @ 图 灵 社 区 : 电子 书 和 好 文章 的 消息 

微 博 @ 图 灵 新 知 : 图 灵 教 育 的 科普 小 组 

Mis 图 灵 访 谈 : ituring_interview， 讲 述 码 农 精彩 人 生 
微 信 图 灵 教 育 : turingbooks 


